From d0fd1f8e1f57bc443309e64da87f4b67c475d1a1 Mon Sep 17 00:00:00 2001
From: Christophe Geuzaine <cgeuzaine@ulg.ac.be>
Date: Wed, 8 Dec 2004 20:01:22 +0000
Subject: [PATCH] upgrade to Netgen 4.4

---
 Mesh/3D_Mesh_Netgen.cpp                    |    6 +-
 Netgen/Makefile                            |   10 +-
 Netgen/NEWS                                |    9 -
 Netgen/README                              |    2 +-
 Netgen/VERSION                             |    2 +-
 Netgen/libsrc/csg/algprim.cpp              |   80 +-
 Netgen/libsrc/csg/algprim.hpp              |    9 +-
 Netgen/libsrc/csg/brick.cpp                |    2 +-
 Netgen/libsrc/csg/brick.hpp                |    3 +-
 Netgen/libsrc/csg/bspline2d.cpp            |    9 +-
 Netgen/libsrc/csg/csg.hpp                  |   17 +-
 Netgen/libsrc/csg/csgeom.cpp               | 1715 +++--
 Netgen/libsrc/csg/csgeom.hpp               |   48 +-
 Netgen/libsrc/csg/csgparser.cpp            |  576 +-
 Netgen/libsrc/csg/edgeflw.cpp              |  515 +-
 Netgen/libsrc/csg/edgeflw.hpp              |   25 +-
 Netgen/libsrc/csg/edgeflw2.cpp             | 1404 ++++
 Netgen/libsrc/csg/edgeflw_new.cpp          | 1553 +++++
 Netgen/libsrc/csg/edgeflw_old.cpp          | 1405 ++++
 Netgen/libsrc/csg/genmesh.cpp              | 1097 ++-
 Netgen/libsrc/csg/identify.cpp             |  173 +-
 Netgen/libsrc/csg/identify.hpp             |    7 +
 Netgen/libsrc/csg/polyhedra.cpp            |  130 +-
 Netgen/libsrc/csg/polyhedra.hpp            |    6 +
 Netgen/libsrc/csg/singularref.cpp          |   59 +-
 Netgen/libsrc/csg/singularref.hpp          |   23 +
 Netgen/libsrc/csg/solid.cpp                | 2132 +++---
 Netgen/libsrc/csg/solid.hpp                |   86 +-
 Netgen/libsrc/csg/specpoin.cpp             | 2056 +++---
 Netgen/libsrc/csg/specpoin.hpp             |   17 +-
 Netgen/libsrc/csg/specpoin_new.cpp         | 1367 ++++
 Netgen/libsrc/csg/specpoin_old.cpp         | 1370 ++++
 Netgen/libsrc/csg/surface.cpp              |   20 +-
 Netgen/libsrc/csg/surface.hpp              |   13 +-
 Netgen/libsrc/csg/triapprox.cpp            |   91 +-
 Netgen/libsrc/csg/triapprox.hpp            |    9 +-
 Netgen/libsrc/general/array.hpp            |   19 +-
 Netgen/libsrc/general/bitarray.hpp         |    4 +-
 Netgen/libsrc/general/dynamicmem.cpp       |    4 +-
 Netgen/libsrc/general/flags.cpp            |    8 +-
 Netgen/libsrc/general/hashtabl.cpp         |    1 +
 Netgen/libsrc/general/hashtabl.hpp         |  273 +-
 Netgen/libsrc/general/moveablemem.cpp      |   16 +-
 Netgen/libsrc/general/moveablemem.hpp      |    4 +-
 Netgen/libsrc/general/mystring.cpp         |    7 +-
 Netgen/libsrc/general/mystring.hpp         |   16 +
 Netgen/libsrc/general/optmem.cpp           |    8 +-
 Netgen/libsrc/general/optmem.hpp           |   20 +-
 Netgen/libsrc/general/sort.cpp             |    1 +
 Netgen/libsrc/general/sort.hpp             |    1 +
 Netgen/libsrc/general/symbolta.hpp         |   68 +-
 Netgen/libsrc/general/table.cpp            |   79 +-
 Netgen/libsrc/general/table.hpp            |   42 +-
 Netgen/libsrc/general/template.hpp         |   46 +-
 Netgen/libsrc/geom2d/genmesh2d.cpp         |   30 +-
 Netgen/libsrc/geom2d/geom2dmesh.cpp        |  123 +-
 Netgen/libsrc/geom2d/spline2d.cpp          |  597 +-
 Netgen/libsrc/geom2d/spline2d.hpp          |   41 +-
 Netgen/libsrc/geom2d/splinegeometry2.cpp   |  208 +-
 Netgen/libsrc/geom2d/splinegeometry2.hpp   |   44 +-
 Netgen/libsrc/gprim/adtree.cpp             | 3473 +++++-----
 Netgen/libsrc/gprim/adtree.hpp             |    8 +-
 Netgen/libsrc/gprim/geom3d.cpp             |   10 +
 Netgen/libsrc/gprim/geom3d.hpp             |    4 +
 Netgen/libsrc/gprim/geomfuncs.hpp          |    9 +-
 Netgen/libsrc/gprim/geomobjects.hpp        |   13 +
 Netgen/libsrc/gprim/geomops.hpp            |   17 +
 Netgen/libsrc/gprim/transform3d.cpp        |    1 +
 Netgen/libsrc/gprim/transform3d.hpp        |   47 +-
 Netgen/libsrc/include/FlexLexer.h          |    2 +-
 Netgen/libsrc/include/mystdlib.h           |    1 +
 Netgen/libsrc/interface/Makefile           |    2 +-
 Netgen/libsrc/interface/nginterface.cpp    |  387 +-
 Netgen/libsrc/interface/nginterface.h      |   29 +-
 Netgen/libsrc/interface/nglib.cpp          |   61 +-
 Netgen/libsrc/interface/nglib.h            |   30 +-
 Netgen/libsrc/interface/readuser.cpp       |   86 +-
 Netgen/libsrc/interface/writeelmer.cpp     |  127 +
 Netgen/libsrc/interface/writefluent.cpp    |   27 +-
 Netgen/libsrc/interface/writegmsh.cpp      |  200 +
 Netgen/libsrc/interface/writepermas.cpp    |  327 +-
 Netgen/libsrc/interface/writepermas2.cpp   |  173 +
 Netgen/libsrc/interface/writetecplot.cpp   |    2 +-
 Netgen/libsrc/interface/writeuser.cpp      |  204 +-
 Netgen/libsrc/interface/writeuser.hpp      |   21 +-
 Netgen/libsrc/interface/wuchemnitz.cpp     |    2 +-
 Netgen/libsrc/linalg/Makefile              |    2 +-
 Netgen/libsrc/linalg/basemat.cpp           |   51 +-
 Netgen/libsrc/linalg/basemat.hpp           |   24 +-
 Netgen/libsrc/linalg/densemat.hpp          |   16 +-
 Netgen/libsrc/linalg/linalg.hpp            |    2 +
 Netgen/libsrc/linalg/sparsmat.cpp          |   60 +-
 Netgen/libsrc/linalg/sparsmat.hpp          |   16 +-
 Netgen/libsrc/linalg/vector.cpp            |    1 +
 Netgen/libsrc/makefile.inc                 |   10 +-
 Netgen/libsrc/makefile.mach.FREEBSD        |   26 +
 Netgen/libsrc/makefile.mach.INTEL          |   22 +-
 Netgen/libsrc/makefile.mach.LINUX          |   21 +-
 Netgen/libsrc/makefile.mach.LINUXGCC33     |   33 +
 Netgen/libsrc/meshing/adfront2.cpp         |   47 +-
 Netgen/libsrc/meshing/adfront2.hpp         |    3 +-
 Netgen/libsrc/meshing/adfront3.cpp         |  150 +-
 Netgen/libsrc/meshing/adfront3.hpp         |   28 +-
 Netgen/libsrc/meshing/bisect.cpp           |   99 +-
 Netgen/libsrc/meshing/bisect.hpp           |    8 +-
 Netgen/libsrc/meshing/clusters.cpp         |   13 +-
 Netgen/libsrc/meshing/curvedelems.cpp      |  145 +-
 Netgen/libsrc/meshing/curvedelems.hpp      |    6 +-
 Netgen/libsrc/meshing/curvedelems2.cpp     |   65 +-
 Netgen/libsrc/meshing/delaunay.cpp         |  471 +-
 Netgen/libsrc/meshing/findip.hpp           |    8 -
 Netgen/libsrc/meshing/geomsearch.cpp       |  434 +-
 Netgen/libsrc/meshing/global.cpp           |    6 +-
 Netgen/libsrc/meshing/hpref_prism.hpp      |  300 +
 Netgen/libsrc/meshing/hpref_quad.hpp       | 2129 ++++++
 Netgen/libsrc/meshing/hpref_tet.hpp        | 2842 ++++++++
 Netgen/libsrc/meshing/hpref_trig.hpp       |  750 ++
 Netgen/libsrc/meshing/hprefinement.cpp     | 5788 +++++++---------
 Netgen/libsrc/meshing/hprefinement.hpp     |  207 +-
 Netgen/libsrc/meshing/improve2.cpp         |   68 +-
 Netgen/libsrc/meshing/improve2.hpp         |    2 +-
 Netgen/libsrc/meshing/improve2gen.cpp      |  348 +-
 Netgen/libsrc/meshing/improve3.cpp         |  181 +-
 Netgen/libsrc/meshing/localh.hpp           |    2 +
 Netgen/libsrc/meshing/meshclass.cpp        | 7247 +++++++++++---------
 Netgen/libsrc/meshing/meshclass.hpp        |   61 +-
 Netgen/libsrc/meshing/meshfunc.cpp         |  117 +-
 Netgen/libsrc/meshing/meshfunc.hpp         |    7 +-
 Netgen/libsrc/meshing/meshing.hpp          |   16 +-
 Netgen/libsrc/meshing/meshing2.cpp         |  302 +-
 Netgen/libsrc/meshing/meshing3.cpp         |  119 +-
 Netgen/libsrc/meshing/meshing3.hpp         |   70 +-
 Netgen/libsrc/meshing/meshtool.cpp         | 1291 ++--
 Netgen/libsrc/meshing/meshtool.hpp         |    1 +
 Netgen/libsrc/meshing/meshtype.cpp         |  571 +-
 Netgen/libsrc/meshing/meshtype.hpp         |  138 +-
 Netgen/libsrc/meshing/msghandler.cpp       |   91 +-
 Netgen/libsrc/meshing/msghandler.hpp       |    9 +-
 Netgen/libsrc/meshing/netrule2.cpp         |   64 +-
 Netgen/libsrc/meshing/netrule3.cpp         |   17 +-
 Netgen/libsrc/meshing/parser2.cpp          |   15 +-
 Netgen/libsrc/meshing/parser3.cpp          |    2 +-
 Netgen/libsrc/meshing/prism2rls.cpp        |   13 +-
 Netgen/libsrc/meshing/prism2rls_2.cpp      |  446 ++
 Netgen/libsrc/meshing/quadrls.cpp          |   60 +-
 Netgen/libsrc/meshing/refine.cpp           |  554 +-
 Netgen/libsrc/meshing/ruler2.cpp           |  136 +-
 Netgen/libsrc/meshing/ruler2.hpp           |   27 +-
 Netgen/libsrc/meshing/ruler3.cpp           |    1 -
 Netgen/libsrc/meshing/secondorder.cpp      |  220 +-
 Netgen/libsrc/meshing/smoothing2.cpp       | 1326 ++--
 Netgen/libsrc/meshing/smoothing3.cpp       |  171 +-
 Netgen/libsrc/meshing/topology.cpp         |  216 +-
 Netgen/libsrc/meshing/topology.hpp         |   10 +-
 Netgen/libsrc/meshing/zrefine.cpp          |    2 +-
 Netgen/libsrc/occ/occgenmesh.cpp           | 1121 ++-
 Netgen/libsrc/occ/occgeom.cpp              | 1072 ++-
 Netgen/libsrc/occ/occgeom.hpp              |  230 +-
 Netgen/libsrc/occ/occmeshsurf.cpp          |  343 +-
 Netgen/libsrc/occ/occmeshsurf.hpp          |   30 +-
 Netgen/libsrc/stlgeom/meshstlsurface.cpp   |   16 +-
 Netgen/libsrc/stlgeom/meshstlsurface.hpp   |    1 +
 Netgen/libsrc/stlgeom/stlgeom.cpp          |    3 +-
 Netgen/libsrc/stlgeom/stlgeom.hpp          |    2 +-
 Netgen/libsrc/stlgeom/stlgeomchart.cpp     |    2 +-
 Netgen/libsrc/stlgeom/stlgeommesh.cpp      |   13 +-
 Netgen/libsrc/visualization/Makefile       |    2 +-
 Netgen/libsrc/visualization/mvdraw.cpp     | 2193 +++---
 Netgen/libsrc/visualization/mvdraw.hpp     |   24 +-
 Netgen/libsrc/visualization/soldata.hpp    |   28 +-
 Netgen/libsrc/visualization/stlmeshing.cpp |   48 +-
 Netgen/libsrc/visualization/vispar.hpp     |   89 +
 Netgen/libsrc/visualization/vscsg.cpp      |    2 +-
 Netgen/libsrc/visualization/vsmesh.cpp     | 2877 ++++----
 Netgen/libsrc/visualization/vsocc.cpp      |  743 ++
 Netgen/libsrc/visualization/vssolution.cpp | 1009 ++-
 Netgen/libsrc/visualization/vssolution.hpp |   22 +-
 Netgen/nglib_addon.cpp                     |   12 +-
 178 files changed, 39979 insertions(+), 20576 deletions(-)
 delete mode 100644 Netgen/NEWS
 create mode 100644 Netgen/libsrc/csg/edgeflw2.cpp
 create mode 100644 Netgen/libsrc/csg/edgeflw_new.cpp
 create mode 100644 Netgen/libsrc/csg/edgeflw_old.cpp
 create mode 100644 Netgen/libsrc/csg/specpoin_new.cpp
 create mode 100644 Netgen/libsrc/csg/specpoin_old.cpp
 create mode 100644 Netgen/libsrc/interface/writeelmer.cpp
 create mode 100644 Netgen/libsrc/interface/writegmsh.cpp
 create mode 100644 Netgen/libsrc/interface/writepermas2.cpp
 create mode 100644 Netgen/libsrc/makefile.mach.FREEBSD
 create mode 100644 Netgen/libsrc/makefile.mach.LINUXGCC33
 create mode 100644 Netgen/libsrc/meshing/hpref_prism.hpp
 create mode 100644 Netgen/libsrc/meshing/hpref_quad.hpp
 create mode 100644 Netgen/libsrc/meshing/hpref_tet.hpp
 create mode 100644 Netgen/libsrc/meshing/hpref_trig.hpp
 create mode 100644 Netgen/libsrc/meshing/prism2rls_2.cpp
 create mode 100644 Netgen/libsrc/visualization/vispar.hpp
 create mode 100644 Netgen/libsrc/visualization/vsocc.cpp

diff --git a/Mesh/3D_Mesh_Netgen.cpp b/Mesh/3D_Mesh_Netgen.cpp
index 47e93fc394..2bf80623de 100644
--- a/Mesh/3D_Mesh_Netgen.cpp
+++ b/Mesh/3D_Mesh_Netgen.cpp
@@ -1,4 +1,4 @@
-// $Id: 3D_Mesh_Netgen.cpp,v 1.11 2004-07-14 22:42:26 geuzaine Exp $
+// $Id: 3D_Mesh_Netgen.cpp,v 1.12 2004-12-08 20:01:21 geuzaine Exp $
 //
 // Copyright (C) 1997-2004 C. Geuzaine, J.-F. Remacle
 //
@@ -53,8 +53,12 @@ void Optimize_Netgen(Mesh * m)
 
 #else
 
+namespace nglib {
 #include "nglib.h"
 #include "nglib_addon.h"
+}
+
+using namespace nglib;
 
 class Netgen{
  private:
diff --git a/Netgen/Makefile b/Netgen/Makefile
index 7b5828793d..0fe37469fa 100644
--- a/Netgen/Makefile
+++ b/Netgen/Makefile
@@ -1,4 +1,4 @@
-# $Id: Makefile,v 1.6 2004-07-03 02:28:46 geuzaine Exp $
+# $Id: Makefile,v 1.7 2004-12-08 20:01:21 geuzaine Exp $
 #
 # Copyright (C) 1997-2004 C. Geuzaine, J.-F. Remacle
 #
@@ -68,6 +68,7 @@ SRC = libsrc/opti/linopt.cpp \
 	libsrc/meshing/clusters.cpp \
 	libsrc/meshing/curvedelems.cpp \
 	libsrc/meshing/curvedelems2.cpp \
+	libsrc/meshing/hprefinement.cpp \
 	libsrc/interface/nglib.cpp \
 	libsrc/gprim/geomtest3d.cpp \
 	libsrc/gprim/geom2d.cpp \
@@ -257,7 +258,7 @@ bisect.o: libsrc/meshing/bisect.cpp libsrc/include/mystdlib.h \
   libsrc/meshing/curvedelems.hpp libsrc/meshing/bisect.hpp \
   libsrc/meshing/clusters.hpp libsrc/meshing/meshfunc.hpp \
   libsrc/meshing/hprefinement.hpp libsrc/meshing/boundarylayer.hpp \
-  libsrc/meshing/specials.hpp libsrc/interface/writeuser.hpp
+  libsrc/meshing/specials.hpp
 meshtool.o: libsrc/meshing/meshtool.cpp libsrc/include/mystdlib.h \
   libsrc/meshing/meshing.hpp libsrc/include/myadt.hpp \
   libsrc/general/myadt.hpp libsrc/include/mydefs.hpp \
@@ -2169,10 +2170,7 @@ genmesh.o: libsrc/csg/genmesh.cpp libsrc/include/mystdlib.h \
   libsrc/csg/explicitcurve2d.hpp libsrc/csg/gencyl.hpp \
   libsrc/csg/polyhedra.hpp libsrc/csg/extrusion.hpp \
   libsrc/csg/revolution.hpp libsrc/csg/specpoin.hpp \
-  libsrc/csg/edgeflw.hpp libsrc/csg/meshsurf.hpp \
-  libsrc/include/stlgeom.hpp libsrc/stlgeom/stlgeom.hpp \
-  libsrc/stlgeom/stltopology.hpp libsrc/stlgeom/stltool.hpp \
-  libsrc/stlgeom/stlline.hpp libsrc/stlgeom/meshstlsurface.hpp
+  libsrc/csg/edgeflw.hpp libsrc/csg/meshsurf.hpp
 spline3d.o: libsrc/csg/spline3d.cpp libsrc/include/mystdlib.h \
   libsrc/include/myadt.hpp libsrc/general/myadt.hpp \
   libsrc/include/mydefs.hpp libsrc/general/ngexception.hpp \
diff --git a/Netgen/NEWS b/Netgen/NEWS
deleted file mode 100644
index 76efa8384b..0000000000
--- a/Netgen/NEWS
+++ /dev/null
@@ -1,9 +0,0 @@
-New features and improvements in NG4.3:
-
-- support for high order curved elements
-- new, hand written CSG parser
-- CSG primitives 'ellipticcylinder' and 'ellipsoid'
-- improvements for Delaunay meshing
-- improvements in surface optimization
-- mesh interface with full topology (element-face-edge-vertex)
-
diff --git a/Netgen/README b/Netgen/README
index 75a8ff1ebf..12dca8d3b7 100644
--- a/Netgen/README
+++ b/Netgen/README
@@ -4,7 +4,7 @@ Sch\"oberl's NETGEN mesh generator:
 
 - only the libsrc directory was kept from the original distribution
 
-- the file meshing/improve2d.cpp was slightly modified to fix build
+- the file meshing/improve2.cpp was slightly modified to fix build
   problems on machines without TCL/TK
 
 **IMPORTANT NOTICE** 
diff --git a/Netgen/VERSION b/Netgen/VERSION
index 74c0c9cbcc..f56504b6db 100644
--- a/Netgen/VERSION
+++ b/Netgen/VERSION
@@ -1 +1 @@
-NG Version 4.3.1
\ No newline at end of file
+NG Version 4.4
\ No newline at end of file
diff --git a/Netgen/libsrc/csg/algprim.cpp b/Netgen/libsrc/csg/algprim.cpp
index 756e4bd1af..1123706b1e 100644
--- a/Netgen/libsrc/csg/algprim.cpp
+++ b/Netgen/libsrc/csg/algprim.cpp
@@ -65,6 +65,7 @@ void QuadraticSurface :: PrintCoeff (ostream & ost) const
 }
 
 
+
 Point<3> QuadraticSurface :: GetSurfacePoint () const
 {
   MyError ("GetSurfacePoint called for QuadraticSurface");
@@ -152,8 +153,8 @@ int Plane :: IsIdentic (const Surface & s2, int & inv, double eps) const
   if (fabs (s2.CalcFunctionValue(hp)) > eps) return 0;
 
   Vec<3> n1, n2;
-  GetNormalVector (p, n1);
-  s2.GetNormalVector (p, n2);
+  n1 = GetNormalVector (p);
+  n2 = s2.GetNormalVector (p);
   inv = (n1 * n2 < 0);
   return 1;
 }
@@ -195,7 +196,7 @@ void Plane :: FromPlane (const Point<2> & pplane, Point<3> & p3d, double h) cons
 
 void Plane :: Project (Point<3> & p3d) const
 {
-  double val = CalcFunctionValue (p3d);
+  double val = Plane::CalcFunctionValue (p3d);
   p3d -= val * n;
 }
 
@@ -205,7 +206,7 @@ INSOLID_TYPE Plane :: BoxInSolid (const BoxSphere<3> & box) const
   double val;
   Point<3> p;
 
-  val = CalcFunctionValue (box.Center());
+  val = Plane::CalcFunctionValue (box.Center());
   if (val > box.Diam() / 2) return IS_OUTSIDE;
   if (val < -box.Diam() / 2) return IS_INSIDE;
 
@@ -230,7 +231,7 @@ INSOLID_TYPE Plane :: BoxInSolid (const BoxSphere<3> & box) const
       for (i = 0; i < 8; i++)
 	{
 	  p = box.GetPointNr (i);
-	  val = CalcFunctionValue (p);
+	  val = Plane::CalcFunctionValue (p);
 	  if (val < 0) 
 	    return DOES_INTERSECT;
 	}
@@ -257,7 +258,7 @@ INSOLID_TYPE Plane :: BoxInSolid (const BoxSphere<3> & box) const
       for (i = 0; i < 8; i++)
 	{
 	  p = box.GetPointNr (i);
-	  val = CalcFunctionValue (p);
+	  val = Plane::CalcFunctionValue (p);
 	  if (val > 0) 
 	    return DOES_INTERSECT;
 	}
@@ -283,10 +284,10 @@ INSOLID_TYPE Plane :: BoxInSolid (const BoxSphere<3> & box) const
 
 
 
-double Plane :: CalcFunctionValue (const Point<3> & p3d) const
-{
-  return cx * p3d(0) + cy * p3d(1) + cz * p3d(2) + c1;
-}
+// double Plane :: CalcFunctionValue (const Point<3> & p3d) const
+// {
+//   return cx * p3d(0) + cy * p3d(1) + cz * p3d(2) + c1;
+// }
 
 void Plane :: CalcGradient (const Point<3> & /* p */, Vec<3> & grad) const
 {
@@ -583,7 +584,6 @@ Ellipsoid (const Point<3> & aa,
   v2 = av2;
   v3 = av3;
 
-  cout << "orhtogonal ? " << (v1*v2) << ", " << (v1*v3) << ", " << (v2*v3) << endl;
   CalcData();
 }
 
@@ -595,11 +595,11 @@ void Ellipsoid :: CalcData ()
   
   Vec<3> hv1, hv2, hv3;
   double lv1 = v1.Length2 ();
-  if (!lv1) lv1 = 1;
+  if (lv1 < 1e-32) lv1 = 1;
   double lv2 = v2.Length2 ();
-  if (!lv2) lv2 = 1;
+  if (lv2 < 1e-32) lv2 = 1;
   double lv3 = v3.Length2 ();
-  if (!lv3) lv3 = 1;
+  if (lv3 < 1e-32) lv3 = 1;
 
   rmin = sqrt (min3 (lv1, lv2, lv3));
 
@@ -888,15 +888,6 @@ void Cylinder :: DefineTangentialPlane (const Point<3> & ap1, const Point<3> & a
   ex /= ex.Length();
 
   ey = Cross (ez, ex);
-
-  /*
-    ez = p1 - a;
-    ez -= (ez * vab) * vap;
-    ez /= ez.Length();
-  
-    ezt = Cross (vab, ez);
-    */
-  
 }
 
 
@@ -1107,9 +1098,9 @@ void EllipticCylinder :: CalcData ()
 
   Vec<3> hvl, hvs;
   double lvl = vl.Length2 ();
-  if (!lvl) lvl = 1;
+  if (lvl < 1e-32) lvl = 1;
   double lvs = vs.Length2 ();
-  if (!lvs) lvs = 1;
+  if (lvs < 1e-32) lvs = 1;
 
   hvl = (1.0 / lvl) * vl;
   hvs = (1.0 / lvs) * vs;
@@ -1255,7 +1246,7 @@ void Cone :: CalcData ()
 
   minr = (ra < rb) ? ra : rb;
 
-  vab = (b - a);
+  vab = b - a;
   vabl = vab.Length();
 
   Vec<3> va (a);
@@ -1265,7 +1256,7 @@ void Cone :: CalcData ()
   //
   //   z(P) = t0vec * P + t0 = (P-a, b-a)/(b-a,b-a)
   //   R(z(P)) = t1vec * P + t1 = rb * z + ra * (1-z)
-  //   r(P)^2 =||P-a||^2 - ||a-b||^2 z^2
+  //   r(P)^2 =||P-a||^2 - ||a-b||^2 z^2k
 
 
   t0vec = vab;
@@ -1293,9 +1284,9 @@ void Cone :: CalcData ()
 
   c1 = va.Length2() - (vab * vab) * t0 * t0 - t1 * t1;
 
-  (*testout) << "t0vec = " << t0vec << " t0 = " << t0 << endl;
-  (*testout) << "t1vec = " << t1vec << " t1 = " << t1 << endl;
-  PrintCoeff (*testout);
+  // (*testout) << "t0vec = " << t0vec << " t0 = " << t0 << endl;
+  // (*testout) << "t1vec = " << t1vec << " t1 = " << t1 << endl;
+  // PrintCoeff (*testout);
 }
 
 
@@ -1316,9 +1307,36 @@ INSOLID_TYPE Cone :: BoxInSolid (const BoxSphere<3> & box) const
 
 double Cone :: HesseNorm () const
 {
-  return 2 / minr;   // old: 2 / minr
+  return 2 / minr;
+}
+
+
+double Cone ::  LocH (const Point<3> & p, double x, 
+				  double c, double hmax) const
+{
+  double bloch = Surface::LocH (p, x, c, hmax);
+  Vec<3> g;
+  CalcGradient (p, g);
+
+  double lam = Abs(g);
+  double meancurv = 
+    -( 2  * g(0)*g(1)*cxy - 2 * czz * (g(0)*g(0)+g(1)*g(1))
+       +2 * g(1)*g(2)*cyz - 2 * cxx * (g(1)*g(1)+g(2)*g(2))
+       +2 * g(0)*g(2)*cxz - 2 * cyy * (g(0)*g(0)+g(2)*g(2))) / (3*lam*lam*lam);
+
+  // cout << "type = " << typeid(*this).name() << ", baseh = " << bloch << ", meancurv = " << meancurv << endl;
+  // return bloch;
+  
+  meancurv = fabs (meancurv);
+  if (meancurv < 1e-20) meancurv = 1e-20;
+
+  // cout << "c = " << c << ", safety = " << mparam.curvaturesafety << endl;
+  double hcurv = 1.0/(3*meancurv*mparam.curvaturesafety);
+
+  return min2 (hmax, hcurv);
 }
 
+
 Point<3> Cone :: GetSurfacePoint () const
 {
   Vec<3> vr = vab.GetNormal ();
diff --git a/Netgen/libsrc/csg/algprim.hpp b/Netgen/libsrc/csg/algprim.hpp
index 83261edd16..aeb829f9c1 100644
--- a/Netgen/libsrc/csg/algprim.hpp
+++ b/Netgen/libsrc/csg/algprim.hpp
@@ -38,8 +38,10 @@ public:
     const { return DOES_INTERSECT; }
 */
   virtual double HesseNorm () const { return cxx + cyy + czz; }
+
   virtual Point<3> GetSurfacePoint () const;
 
+
   virtual void Print (ostream & ist) const;
   virtual void Read (istream & ist);
   void PrintCoeff (ostream & ost) const;
@@ -86,7 +88,8 @@ public:
   virtual INSOLID_TYPE BoxInSolid (const BoxSphere<3> & box) const;
 
   ///
-  virtual double CalcFunctionValue (const Point<3> & point) const;
+  inline virtual double CalcFunctionValue (const Point<3> & p3d) const
+  {return cx * p3d(0) + cy * p3d(1) + cz * p3d(2) + c1;}
   ///
   virtual void CalcGradient (const Point<3> & point, 
 			     Vec<3> & grad) const;
@@ -316,6 +319,10 @@ public:
   virtual INSOLID_TYPE BoxInSolid (const BoxSphere<3> & box) const;
   ///
   virtual double HesseNorm () const;
+
+  virtual double LocH (const Point<3> & p, double x, 
+		       double c, double hmax) const;
+
   ///
   virtual Point<3> GetSurfacePoint () const;
 
diff --git a/Netgen/libsrc/csg/brick.cpp b/Netgen/libsrc/csg/brick.cpp
index e4224ae512..f408e922f4 100644
--- a/Netgen/libsrc/csg/brick.cpp
+++ b/Netgen/libsrc/csg/brick.cpp
@@ -52,7 +52,7 @@ IsIdentic (const Surface & s2, int & inv, double eps) const
   if (id)
     {
       Vec<3> n2;
-      s2.GetNormalVector(p1, n2);
+      n2 = s2.GetNormalVector(p1);
       inv = (n * n2) < 0;
     }
   return id;
diff --git a/Netgen/libsrc/csg/brick.hpp b/Netgen/libsrc/csg/brick.hpp
index 1733335101..11106b66d3 100644
--- a/Netgen/libsrc/csg/brick.hpp
+++ b/Netgen/libsrc/csg/brick.hpp
@@ -54,7 +54,7 @@ class Brick : public Primitive
 
 public:
   Brick (Point<3> ap1, Point<3> ap2, Point<3> ap3, Point<3> ap4);
-  ~Brick ();
+  virtual ~Brick ();
   static Primitive * CreateDefault ();
 
 
@@ -90,6 +90,7 @@ protected:
   Point<3> pmin, pmax;
 public:
   OrthoBrick (const Point<3> & ap1, const Point<3> & ap2);
+  
   virtual INSOLID_TYPE BoxInSolid (const BoxSphere<3> & box) const;
   virtual void Reduce (const BoxSphere<3> & box);
 };
diff --git a/Netgen/libsrc/csg/bspline2d.cpp b/Netgen/libsrc/csg/bspline2d.cpp
index ebb284a28b..93b5e89621 100644
--- a/Netgen/libsrc/csg/bspline2d.cpp
+++ b/Netgen/libsrc/csg/bspline2d.cpp
@@ -30,14 +30,13 @@ bool BSplineCurve2d :: Inside (const Point<2> & p, double & dist) const
   dist = Dist (p, hp);
   double scal = (hp-p) * n;
   cout << "scal = " << scal << endl;
-  char ch;
-  //  cin >> ch;
+
   return scal >= 0;
 }
   
 double BSplineCurve2d :: ProjectParam (const Point<2> & p) const
 {
-  double t, dt, mindist, mint;
+  double t, dt, mindist, mint = 0.0;
   int n1;
   
   mindist = 1e10;
@@ -148,11 +147,11 @@ Vec<2> BSplineCurve2d :: EvalPrime (double t) const
 Vec<2> BSplineCurve2d :: EvalPrimePrime (double t) const
 {
   int n, n1, n2, n3, n4;
-  double loct, ddb1, ddb2, ddb3, ddb4;
+  double ddb1, ddb2, ddb3, ddb4;
   Vec<2> hv;
   
   n = int(t);   
-  //  loct = t - n;
+  //  double loct = t - n;
   
   ddb1 = 0.5;
   ddb4 = 0.5;
diff --git a/Netgen/libsrc/csg/csg.hpp b/Netgen/libsrc/csg/csg.hpp
index 08dd054901..e150860232 100644
--- a/Netgen/libsrc/csg/csg.hpp
+++ b/Netgen/libsrc/csg/csg.hpp
@@ -17,15 +17,21 @@ namespace netgen
 #include "solid.hpp"
 #include "identify.hpp"
 #include "singularref.hpp"
-
 #include "csgeom.hpp"
-#include "triapprox.hpp"
 
+#ifndef SMALLLIB
+#define _INCLUDE_MORE
+#endif
+#ifdef LINUX
+#define _INCLUDE_MORE
+#endif
+
+#ifdef _INCLUDE_MORE
+#include "triapprox.hpp"
 
 #include "algprim.hpp"
 #include "brick.hpp"
 #include "spline3d.hpp"
-// #include "spline2d.hpp"
 #include "manifold.hpp"
 #include "curve2d.hpp"
 #include "explicitcurve2d.hpp"
@@ -33,11 +39,10 @@ namespace netgen
 #include "polyhedra.hpp"
 #include "extrusion.hpp"
 #include "revolution.hpp"
-
-// #include "geom2dmesh.hpp"
 #include "specpoin.hpp"
 #include "edgeflw.hpp"
-#include "meshsurf.hpp" 
+#include "meshsurf.hpp"
+#endif
 }
 
 #endif
diff --git a/Netgen/libsrc/csg/csgeom.cpp b/Netgen/libsrc/csg/csgeom.cpp
index 7e1051b781..f7c25ee9c6 100644
--- a/Netgen/libsrc/csg/csgeom.cpp
+++ b/Netgen/libsrc/csg/csgeom.cpp
@@ -8,359 +8,355 @@
 namespace netgen
 {
 
-int CSGeometry :: changeval = 0;
+  int CSGeometry :: changeval = 0;
 
 
 
-TopLevelObject ::  
-TopLevelObject (Solid * asolid,
-		Surface * asurface)
-{
-  solid = asolid;
-  surface = asurface;
-
-  SetRGB (0, 0, 1);
-  SetTransparent (0);
-  SetVisible (1); 
-  SetLayer (1);
-
-  if (!surface)
-    maxh = solid->GetMaxH();
-  else
-    maxh = surface->GetMaxH();
-
-  SetBCProp (-1);
-}
-
-void TopLevelObject :: GetData (ostream & ost)
-{
-  ost << red << " " << green << " " << blue << " " 
-      << transp << " " << visible << " ";
-}
-
-void TopLevelObject :: SetData (istream & ist)
-{
-  ist >> red >> green >> blue >> transp >> visible;
-}
-
-
+  TopLevelObject ::  
+  TopLevelObject (Solid * asolid,
+		  Surface * asurface)
+  {
+    solid = asolid;
+    surface = asurface;
 
+    SetRGB (0, 0, 1);
+    SetTransparent (0);
+    SetVisible (1); 
+    SetLayer (1);
 
+    if (!surface)
+      maxh = solid->GetMaxH();
+    else
+      maxh = surface->GetMaxH();
 
-CSGeometry :: CSGeometry ()
-  : boundingbox (Point<3> (-1000, -1000, -1000),
-		 Point<3> (1000, 1000, 1000)),
-    identicsurfaces (100), filename("")
-{
-  ;
-}
-
+    SetBCProp (-1);
+  }
 
+  void TopLevelObject :: GetData (ostream & ost)
+  {
+    ost << red << " " << green << " " << blue << " " 
+	<< transp << " " << visible << " ";
+  }
 
-CSGeometry :: CSGeometry (const string & afilename)
-  : boundingbox (Point<3> (-1000, -1000, -1000),
-		 Point<3> (1000, 1000, 1000)),
-    identicsurfaces (100), filename(afilename)
-{
-  changeval++;
-}
+  void TopLevelObject :: SetData (istream & ist)
+  {
+    ist >> red >> green >> blue >> transp >> visible;
+  }
 
-CSGeometry :: ~CSGeometry ()
-{
-  Clean();
-}
 
-
-void CSGeometry :: Clean ()
-{
-  int i;
-
-  //  for (i = 0; i < solids.Size(); i++)
-  //    delete solids[i];
-  solids.DeleteAll ();
-
-  for (i = 0; i < surfaces.Size(); i++)
-    delete surfaces[i];
-  surfaces.DeleteAll ();
+ 
+  Box<3> CSGeometry::default_boundingbox (Point<3> (-1000, -1000, -1000),
+					  Point<3> ( 1000,  1000,  1000));
+
+
+  CSGeometry :: CSGeometry ()
+    : boundingbox (default_boundingbox),
+      identicsurfaces (100), filename("")
+  {
+    ;
+  }
+
+  CSGeometry :: CSGeometry (const string & afilename)
+    : boundingbox (default_boundingbox),
+      identicsurfaces (100), filename(afilename)
+  {
+    changeval++;
+  }
+
+  CSGeometry :: ~CSGeometry ()
+  {
+    Clean();
+  }
+
+
+  void CSGeometry :: Clean ()
+  {
+    for (int i = 0; i < solids.Size(); i++)
+      delete solids[i]->S1();
+    for (int i = 0; i < solids.Size(); i++)
+      delete solids[i];
+    solids.DeleteAll ();
+
+    /*
+    for (int i = 0; i < surfaces.Size(); i++)
+      delete surfaces[i];
+    surfaces.DeleteAll ();
+    */
   
-  for (i = 0; i < toplevelobjects.Size(); i++)
-    delete toplevelobjects[i];
-  toplevelobjects.DeleteAll ();
+    for (int i = 0; i < toplevelobjects.Size(); i++)
+      delete toplevelobjects[i];
+    toplevelobjects.DeleteAll ();
+    for (int i = 0; i < triapprox.Size(); i++)
+      delete triapprox[i];
+    triapprox.DeleteAll();
 
-  for (i = 0; i < triapprox.Size(); i++)
-    delete triapprox[i];
-  triapprox.DeleteAll();
+    changeval++;
+  }
 
-  changeval++;
-}
 
 
 
 
+  class WritePrimitivesIt : public SolidIterator
+  {
+    ostream & ost;
+  public:
+    WritePrimitivesIt (ostream & aost) : ost(aost) { ; }
+    virtual ~WritePrimitivesIt () { ; }
 
-class WritePrimitivesIt : public SolidIterator
-{
-  ostream & ost;
-public:
-  WritePrimitivesIt (ostream & aost);
-  ~WritePrimitivesIt ();
-  virtual void Do (Solid * sol);
-};
-
-WritePrimitivesIt :: WritePrimitivesIt (ostream & aost)
-  : SolidIterator(), ost(aost)
-{
-  ;
-}
-WritePrimitivesIt :: ~WritePrimitivesIt ()
-{
-  ;
-}
+    virtual void Do (Solid * sol);
+  };
 
-void WritePrimitivesIt :: Do (Solid * sol) 
-{
-  Primitive * prim = sol->GetPrimitive();
-  if (prim)
-    {
-      char * classname;
-      ARRAY<double> coeffs;
-
-      prim -> GetPrimitiveData (classname, coeffs);
-
-      if (sol->Name())
-	ost << "primitive " 
-	    << sol->Name() << " "
-	    << classname << "  " << coeffs.Size();
-      for (int i = 0; i < coeffs.Size(); i++)
-	ost << " " << coeffs[i];
-      ost << endl;
-    }
-}
+  void WritePrimitivesIt :: Do (Solid * sol) 
+  {
+    Primitive * prim = sol->GetPrimitive();
+    if (prim)
+      {
+	char * classname;
+	ARRAY<double> coeffs;
+
+	prim -> GetPrimitiveData (classname, coeffs);
+
+	if (sol->Name())
+	  ost << "primitive " 
+	      << sol->Name() << " "
+	      << classname << "  " << coeffs.Size();
+	for (int i = 0; i < coeffs.Size(); i++)
+	  ost << " " << coeffs[i];
+	ost << endl;
+      }
+  }
 
+  
+  void CSGeometry :: Save (ostream & ost) 
+  {
+    ost << "boundingbox "
+	<< boundingbox.PMin()(0) << " "
+	<< boundingbox.PMin()(1) << " "
+	<< boundingbox.PMin()(2) << " "
+	<< boundingbox.PMax()(0) << " "
+	<< boundingbox.PMax()(1) << " "
+	<< boundingbox.PMax()(2) << endl;
 
 
+    WritePrimitivesIt wpi(ost);
+    IterateAllSolids (wpi, 1);
 
-void CSGeometry :: Save (ostream & ost) 
-{
-  ost << "boundingbox "
-      << boundingbox.PMin()(0) << " "
-      << boundingbox.PMin()(1) << " "
-      << boundingbox.PMin()(2) << " "
-      << boundingbox.PMax()(0) << " "
-      << boundingbox.PMax()(1) << " "
-      << boundingbox.PMax()(2) << endl;
-
-
-  WritePrimitivesIt wpi(ost);
-  IterateAllSolids (wpi, 1);
-
-  int i;
-  for (i = 0; i < solids.Size(); i++)
-    {
-      if (!solids[i]->GetPrimitive())
-	{
-	  ost << "solid " << solids.GetName(i) << " ";
-	  solids[i] -> GetSolidData (ost);
-	  ost << endl;
-	}
-    }
+    for (int i = 0; i < solids.Size(); i++)
+      {
+	if (!solids[i]->GetPrimitive())
+	  {
+	    ost << "solid " << solids.GetName(i) << " ";
+	    solids[i] -> GetSolidData (ost);
+	    ost << endl;
+	  }
+      }
 
-  for (i = 0; i < GetNTopLevelObjects(); i++)
-    {
-      TopLevelObject * tlo = GetTopLevelObject (i);
-      ost << "toplevel ";
-      if (tlo -> GetSurface())
-	ost << "surface " << tlo->GetSolid()->Name() << " "
-	    << tlo->GetSurface()->Name() << " ";
-      else
-	ost << "solid " << tlo->GetSolid()->Name() << " ";
-      tlo->GetData(ost);
-      ost << endl;
-    }
+    for (int i = 0; i < GetNTopLevelObjects(); i++)
+      {
+	TopLevelObject * tlo = GetTopLevelObject (i);
+	ost << "toplevel ";
+	if (tlo -> GetSurface())
+	  ost << "surface " << tlo->GetSolid()->Name() << " "
+	      << tlo->GetSurface()->Name() << " ";
+	else
+	  ost << "solid " << tlo->GetSolid()->Name() << " ";
+	tlo->GetData(ost);
+	ost << endl;
+      }
 
-  for (i = 0; i < identifications.Size(); i++)
-    {
-      ost << "identify ";
-      identifications[i] -> GetData (ost);
-      ost << endl;
-    }
+    for (int i = 0; i < identifications.Size(); i++)
+      {
+	ost << "identify ";
+	identifications[i] -> GetData (ost);
+	ost << endl;
+      }
 
-  ost << "end" << endl;
-}
+    ost << "end" << endl;
+  }
 
  
-void CSGeometry :: Load (istream & ist)
-{
-  //  CSGeometry * geo = new CSGeometry;
+  void CSGeometry :: Load (istream & ist)
+  {
+    //  CSGeometry * geo = new CSGeometry;
   
-  char key[100], name[100], classname[100], sname[100];
-  int ncoeff, i, j;
-  ARRAY<double> coeff;
-
-  while (ist.good())
-    {
-      ist >> key;
-      if (strcmp (key, "boundingbox") == 0)
-	{
-	  Point<3> pmin, pmax;
-	  ist >> pmin(0) >> pmin(1) >> pmin(2);
-	  ist >> pmax(0) >> pmax(1) >> pmax(2);
-	  SetBoundingBox (Box<3> (pmin, pmax));
-	}
-      if (strcmp (key, "primitive") == 0)
-	{
-	  ist >> name >> classname >> ncoeff;
-	  coeff.SetSize (ncoeff);
-	  for (i = 0; i < ncoeff; i++)
-	    ist >> coeff[i];
-
-	  Primitive * nprim = Primitive::CreatePrimitive (classname);
-	  nprim -> SetPrimitiveData (coeff);
-	  Solid * nsol = new Solid (nprim);
-
-	  for (j = 0; j < nprim->GetNSurfaces(); j++)
-	    {
-	      sprintf (sname, "%s,%d", name, j);
-	      AddSurface (sname, &nprim->GetSurface(j));
-	      nprim -> SetSurfaceId (j, GetNSurf());
-	    }
-	  SetSolid (name, nsol);
-	}
-      else if (strcmp (key, "solid") == 0)
-	{
-	  ist >> name;
-	  Solid * nsol = Solid::CreateSolid (ist, solids);
-
-	  cout << " I have found solid " << name << " = ";
-	  nsol -> GetSolidData (cout);
-	  cout << endl;
-
-	  SetSolid (name, nsol);
-	}
-      else if (strcmp (key, "toplevel") == 0)
-	{
-	  char type[20], solname[50], surfname[50];
-	  const Solid * sol = NULL;
-	  const Surface * surf = NULL;
-	  int nr;
+    char key[100], name[100], classname[100], sname[100];
+    int ncoeff, i, j;
+    ARRAY<double> coeff;
 
-	  ist >> type;
-	  if (strcmp (type, "solid") == 0)
-	    {
-	      ist >> solname;
-	      sol = GetSolid (solname);
-	    }
-	  if (strcmp (type, "surface") == 0)
-	    {
-	      ist >> solname >> surfname;
-	      sol = GetSolid (solname);
-	      surf = GetSurface (surfname);
-	    }
-	  nr = SetTopLevelObject ((Solid*)sol, (Surface*)surf);
-	  GetTopLevelObject (nr) -> SetData (ist);
-	}
-      else if (strcmp (key, "identify") == 0)
-	{
-	  char type[10], surfname1[50], surfname2[50];
-	  const Surface * surf1;
-	  const Surface * surf2;
-
-
-	  ist >> type >> surfname1 >> surfname2;
-	  surf1 = GetSurface(surfname1);
-	  surf2 = GetSurface(surfname2);
+    while (ist.good())
+      {
+	ist >> key;
+	if (strcmp (key, "boundingbox") == 0)
+	  {
+	    Point<3> pmin, pmax;
+	    ist >> pmin(0) >> pmin(1) >> pmin(2);
+	    ist >> pmax(0) >> pmax(1) >> pmax(2);
+	    SetBoundingBox (Box<3> (pmin, pmax));
+	  }
+	if (strcmp (key, "primitive") == 0)
+	  {
+	    ist >> name >> classname >> ncoeff;
+	    coeff.SetSize (ncoeff);
+	    for (i = 0; i < ncoeff; i++)
+	      ist >> coeff[i];
+
+	    Primitive * nprim = Primitive::CreatePrimitive (classname);
+	    nprim -> SetPrimitiveData (coeff);
+	    Solid * nsol = new Solid (nprim);
+
+	    for (j = 0; j < nprim->GetNSurfaces(); j++)
+	      {
+		sprintf (sname, "%s,%d", name, j);
+		AddSurface (sname, &nprim->GetSurface(j));
+		nprim -> SetSurfaceId (j, GetNSurf());
+	      }
+	    SetSolid (name, nsol);
+	  }
+	else if (strcmp (key, "solid") == 0)
+	  {
+	    ist >> name;
+	    Solid * nsol = Solid::CreateSolid (ist, solids);
+
+	    cout << " I have found solid " << name << " = ";
+	    nsol -> GetSolidData (cout);
+	    cout << endl;
+
+	    SetSolid (name, nsol);
+	  }
+	else if (strcmp (key, "toplevel") == 0)
+	  {
+	    char type[20], solname[50], surfname[50];
+	    const Solid * sol = NULL;
+	    const Surface * surf = NULL;
+	    int nr;
+
+	    ist >> type;
+	    if (strcmp (type, "solid") == 0)
+	      {
+		ist >> solname;
+		sol = GetSolid (solname);
+	      }
+	    if (strcmp (type, "surface") == 0)
+	      {
+		ist >> solname >> surfname;
+		sol = GetSolid (solname);
+		surf = GetSurface (surfname);
+	      }
+	    nr = SetTopLevelObject ((Solid*)sol, (Surface*)surf);
+	    GetTopLevelObject (nr) -> SetData (ist);
+	  }
+	else if (strcmp (key, "identify") == 0)
+	  {
+	    char type[10], surfname1[50], surfname2[50];
+	    const Surface * surf1;
+	    const Surface * surf2;
+
+
+	    ist >> type >> surfname1 >> surfname2;
+	    surf1 = GetSurface(surfname1);
+	    surf2 = GetSurface(surfname2);
 	  
-	  AddIdentification (new PeriodicIdentification 
-			     (GetNIdentifications(),
-			      *this, surf1, surf2));
-	}
-      else if (strcmp (key, "end") == 0)
-	break;
-    }
+	    AddIdentification (new PeriodicIdentification 
+			       (GetNIdentifications(),
+				*this, surf1, surf2));
+	  }
+	else if (strcmp (key, "end") == 0)
+	  break;
+      }
 
-  changeval++;
-}
+    changeval++;
+  }
 
 
 
 
 
-void CSGeometry :: AddSurface (Surface * surf)
-{
-  static int cntsurfs = 0;
-  cntsurfs++;
-  char name[15];
-  sprintf (name, "nnsurf%d", cntsurfs);
-  AddSurface (name, surf);
-}
+  void CSGeometry :: AddSurface (Surface * surf)
+  {
+    static int cntsurfs = 0;
+    cntsurfs++;
+    char name[15];
+    sprintf (name, "nnsurf%d", cntsurfs);
+    AddSurface (name, surf);
+  }
  
-void CSGeometry :: AddSurface (char * name, Surface * surf)
-{ 
-  surfaces.Set (name, surf); 
-  surf->SetName (name);
-  changeval++; 
-}
-
-
-const Surface * CSGeometry :: GetSurface (const char * name) const
-{
-  if (surfaces.Used(name))
-    return surfaces.Get(name);
-  else
-    return NULL;
-}
+  void CSGeometry :: AddSurface (char * name, Surface * surf)
+  { 
+    surfaces.Set (name, surf); 
+    surf->SetName (name);
+    changeval++; 
+  }
+
+  void CSGeometry :: AddSurfaces (Primitive * prim)
+  {
+    for (int i = 0; i < prim->GetNSurfaces(); i++)
+      {
+	AddSurface (&prim->GetSurface(i));
+	prim->SetSurfaceId (i, GetNSurf()-1);
+      }
+  }
 
-const Surface * CSGeometry :: GetSurface (int i) const
-{
-  if (i >= 0 && i < surfaces.Size()) 
-    return surfaces[i];
-  else
-    return NULL;
-}
+  const Surface * CSGeometry :: GetSurface (const char * name) const
+  {
+    if (surfaces.Used(name))
+      return surfaces.Get(name);
+    else
+      return NULL;
+  }
 
+  /*
+  const Surface * CSGeometry :: GetSurface (int i) const
+  {
+    if (i >= 0 && i < surfaces.Size()) 
+      return surfaces[i];
+    else
+      throw NgException ("CSGeometry::GetSurface out of range");
+  }
+  */
 
 
 
 
-void CSGeometry :: SetSolid (const char * name, Solid * sol)
-{
-  Solid * oldsol = NULL;
+  void CSGeometry :: SetSolid (const char * name, Solid * sol)
+  {
+    Solid * oldsol = NULL;
 
-  if (solids.Used (name))
-    oldsol = solids.Get(name);
+    if (solids.Used (name))
+      oldsol = solids.Get(name);
 
-  solids.Set (name, sol);
-  sol->SetName (name);
+    solids.Set (name, sol);
+    sol->SetName (name);
 
-  if (oldsol)
-    {
-      if (oldsol->op != Solid::ROOT ||
-	  sol->op != Solid::ROOT)
-	{
-	  cerr << "Setsolid: old or new no root" << endl;
-	}
-      oldsol -> s1 = sol -> s1;
-    }
-  changeval++;
-}
+    if (oldsol)
+      {
+	if (oldsol->op != Solid::ROOT ||
+	    sol->op != Solid::ROOT)
+	  {
+	    cerr << "Setsolid: old or new no root" << endl;
+	  }
+	oldsol -> s1 = sol -> s1;
+      }
+    changeval++;
+  }
 
-const Solid * CSGeometry :: GetSolid (const char * name) const
-{
-  if (solids.Used(name))
-    return solids.Get(name);
-  else
-    return NULL;
-}
+  const Solid * CSGeometry :: GetSolid (const char * name) const
+  {
+    if (solids.Used(name))
+      return solids.Get(name);
+    else
+      return NULL;
+  }
 
 
 
-const Solid * CSGeometry :: GetSolid (const string & name) const
-{
-  if (solids.Used(name.c_str()))
-    return solids.Get(name.c_str());
-  else
-    return NULL;
-}
+  const Solid * CSGeometry :: GetSolid (const string & name) const
+  {
+    if (solids.Used(name.c_str()))
+      return solids.Get(name.c_str());
+    else
+      return NULL;
+  }
 
 
 
@@ -369,699 +365,656 @@ const Solid * CSGeometry :: GetSolid (const string & name) const
 
 
 
-class RemoveDummyIterator : public SolidIterator
-{
-public:
+  class RemoveDummyIterator : public SolidIterator
+  {
+  public:
   
-  RemoveDummyIterator();
-  ~RemoveDummyIterator();
-  virtual void Do(Solid * sol);
-};
+    RemoveDummyIterator() { ; }
+    virtual ~RemoveDummyIterator() { ; }
+    virtual void Do(Solid * sol);
+  };
 
-RemoveDummyIterator :: RemoveDummyIterator()
-{
-  ;
-}
-
-RemoveDummyIterator :: ~RemoveDummyIterator()
-{
-  ;
-}
-
-void RemoveDummyIterator :: Do(Solid * sol)
-{
-  if ( (sol->op == Solid::SUB || sol->op == Solid::SECTION || 
-	sol->op == Solid::UNION)
-       && sol->s1->op == Solid::DUMMY)
-    sol->s1 = sol->s1->s1;
-  if ( (sol->op == Solid::SECTION || sol->op == Solid::UNION)
-       && sol->s2->op == Solid::DUMMY)
-    sol->s2 = sol->s2->s1;
-}
+  void RemoveDummyIterator :: Do(Solid * sol)
+  {
+    if ( (sol->op == Solid::SUB || sol->op == Solid::SECTION || 
+	  sol->op == Solid::UNION)
+	 && sol->s1->op == Solid::DUMMY)
+      sol->s1 = sol->s1->s1;
+    if ( (sol->op == Solid::SECTION || sol->op == Solid::UNION)
+	 && sol->s2->op == Solid::DUMMY)
+      sol->s2 = sol->s2->s1;
+  }
 
 
 
 
 
 
-int CSGeometry :: SetTopLevelObject (Solid * sol, Surface * surf)
-{
-  return toplevelobjects.Append (new TopLevelObject (sol, surf)) - 1;
-}
-
-void CSGeometry :: GetTopLevelObject (int nr, Solid *& sol, Surface *& surf)
-{
-  sol = toplevelobjects[nr]->GetSolid();
-  surf = toplevelobjects[nr]->GetSurface();
-}
+  int CSGeometry :: SetTopLevelObject (Solid * sol, Surface * surf)
+  {
+    return toplevelobjects.Append (new TopLevelObject (sol, surf)) - 1;
+  }
 
+  TopLevelObject * CSGeometry :: 
+  GetTopLevelObject (const Solid * sol, const Surface * surf)
+  {
+    for (int i = 0; i < toplevelobjects.Size(); i++)
+      {
+	if (toplevelobjects[i]->GetSolid() == sol &&
+	    toplevelobjects[i]->GetSurface() == surf)
+	  return (toplevelobjects[i]);
+      }
+    return NULL;
+  }
 
-void CSGeometry :: GetTopLevelObject (int nr, const Solid *& sol, const Surface *& surf) const
-{
-  sol = toplevelobjects[nr]->GetSolid();
-  surf = toplevelobjects[nr]->GetSurface();
-}
-
-TopLevelObject * CSGeometry :: 
-GetTopLevelObject (const Solid * sol, const Surface * surf)
-{
-  for (int i = 0; i < toplevelobjects.Size(); i++)
-    {
-      if (toplevelobjects[i]->GetSolid() == sol &&
-	  toplevelobjects[i]->GetSurface() == surf)
-	return (toplevelobjects[i]);
-    }
-  return NULL;
-}
-
-void CSGeometry :: RemoveTopLevelObject (Solid * sol, Surface * surf)
-{
-  for (int i = 0; i < toplevelobjects.Size(); i++)
-    {
-      if (toplevelobjects[i]->GetSolid() == sol &&
-	  toplevelobjects[i]->GetSurface() == surf)
-	{
-	  delete toplevelobjects[i];
-	  toplevelobjects.DeleteElement (i+1);
-	  changeval++;
-	  break;
-	}
-    }
-}
+  void CSGeometry :: RemoveTopLevelObject (Solid * sol, Surface * surf)
+  {
+    for (int i = 0; i < toplevelobjects.Size(); i++)
+      {
+	if (toplevelobjects[i]->GetSolid() == sol &&
+	    toplevelobjects[i]->GetSurface() == surf)
+	  {
+	    delete toplevelobjects[i];
+	    toplevelobjects.DeleteElement (i+1);
+	    changeval++;
+	    break;
+	  }
+      }
+  }
 
-void CSGeometry :: AddIdentification (Identification * ident)
-{
-  identifications.Append (ident);
-}
+  void CSGeometry :: AddIdentification (Identification * ident)
+  {
+    identifications.Append (ident);
+  }
 
-void CSGeometry :: SetFlags (const char * solidname, const Flags & flags)
-{
-  Solid * solid = solids.Elem(solidname);
-  ARRAY<int> surfind;
+  void CSGeometry :: SetFlags (const char * solidname, const Flags & flags)
+  {
+    Solid * solid = solids.Elem(solidname);
+    ARRAY<int> surfind;
 
-  int i;
-  double maxh = flags.GetNumFlag ("maxh", -1);
-  if (maxh > 0 && solid)
-    {
-      solid->GetSurfaceIndices (surfind);
+    int i;
+    double maxh = flags.GetNumFlag ("maxh", -1);
+    if (maxh > 0 && solid)
+      {
+	solid->GetSurfaceIndices (surfind);
 
-      for (i = 0; i < surfind.Size(); i++)
-	{
-	  if (surfaces[surfind[i]]->GetMaxH() > maxh)
-	    surfaces[surfind[i]] -> SetMaxH (maxh);
-	}
+	for (i = 0; i < surfind.Size(); i++)
+	  {
+	    if (surfaces[surfind[i]]->GetMaxH() > maxh)
+	      surfaces[surfind[i]] -> SetMaxH (maxh);
+	  }
 
-      solid->SetMaxH (maxh);
-    }
+	solid->SetMaxH (maxh);
+      }
 
-  if (flags.NumFlagDefined ("bc"))
-    {
-      solid->GetSurfaceIndices (surfind);
-      int bc = int (flags.GetNumFlag("bc", -1));
-      for (i = 0; i < surfind.Size(); i++)
-	{
-	  if (surfaces[surfind[i]]->GetBCProperty() == -1)
-	    surfaces[surfind[i]]->SetBCProperty(bc);
-	}
-    }
-}
+    if (flags.NumFlagDefined ("bc"))
+      {
+	solid->GetSurfaceIndices (surfind);
+	int bc = int (flags.GetNumFlag("bc", -1));
+	for (i = 0; i < surfind.Size(); i++)
+	  {
+	    if (surfaces[surfind[i]]->GetBCProperty() == -1)
+	      surfaces[surfind[i]]->SetBCProperty(bc);
+	  }
+      }
+  }
 
-void CSGeometry :: FindIdenticSurfaces (double eps)
-{
-  int i, j;
-  int inv;
-  int nsurf = GetNSurf();
+  void CSGeometry :: FindIdenticSurfaces (double eps)
+  {
+    int inv;
+    int nsurf = GetNSurf();
 
-  isidenticto.SetSize(nsurf);
-  for (i = 0; i < nsurf; i++)
-    isidenticto[i] = i;
+    isidenticto.SetSize(nsurf);
+    for (int i = 0; i < nsurf; i++)
+      isidenticto[i] = i;
   
-  for (i = 0; i < nsurf; i++)
-    for (j = i+1; j < nsurf; j++)
-      if (GetSurface(j) -> IsIdentic (*GetSurface(i), inv, eps))
-	{
-	  INDEX_2 i2(i, j);
-	  identicsurfaces.Set (i2, inv);
-	  isidenticto[j] = isidenticto[i];
-	  // (*testout) << "surfaces " << i2 << " are identic" << endl;
-	}
-
-  /*
-  (*testout) << "identicmap:" << endl;
-  for (i = 0; i < isidenticto.Size(); i++)
-    (*testout) << i << " -> " << isidenticto[i] << endl;
-
-  for (i = 0; i < nsurf; i++)
-    GetSurface(i)->Print (*testout);
-  */
-}
-
-
-void CSGeometry ::
-GetIndependentSurfaceIndices (const Solid * sol, 
-			      const BoxSphere<3> & box, 
-			      ARRAY<int> & locsurf) const
-{
-  int i, j;
-
-  ReducePrimitiveIterator rpi(box);
-  UnReducePrimitiveIterator urpi;
-
-  ((Solid*)sol) -> IterateSolid (rpi);
-  sol -> GetSurfaceIndices (locsurf);
-  ((Solid*)sol) -> IterateSolid (urpi);
+    for (int i = 0; i < nsurf; i++)
+      for (int j = i+1; j < nsurf; j++)
+	if (GetSurface(j) -> IsIdentic (*GetSurface(i), inv, eps))
+	  {
+	    INDEX_2 i2(i, j);
+	    identicsurfaces.Set (i2, inv);
+	    isidenticto[j] = isidenticto[i];
+	    // (*testout) << "surfaces " << i2 << " are identic" << endl;
+	  }
+
+    /*  
+    (*testout) << "identicmap:" << endl;
+    for (int i = 0; i < isidenticto.Size(); i++)
+      (*testout) << i << " -> " << isidenticto[i] << endl;
+    
+    for (int i = 0; i < nsurf; i++)
+      GetSurface(i)->Print (*testout);
+    */
+  }
   
-  int cntindep = 0;
-  bool indep;
-  for (i = locsurf.Size()-1; i >= 0; i--)
-    {
-      indep = 1;
-      for (j = 0; j < i; j++)
-	{
-	  INDEX_2 i2(locsurf[i], locsurf[j]);
-	  i2.Sort();
-	  if (identicsurfaces.Used (i2))
+  
+  void CSGeometry ::
+  GetIndependentSurfaceIndices (const Solid * sol, 
+				const BoxSphere<3> & box, 
+				ARRAY<int> & locsurf) const
+  {
+    ReducePrimitiveIterator rpi(box);
+    UnReducePrimitiveIterator urpi;
+
+    ((Solid*)sol) -> IterateSolid (rpi);
+    sol -> GetSurfaceIndices (locsurf);
+    ((Solid*)sol) -> IterateSolid (urpi);
+
+    for (int i = 0; i < locsurf.Size(); i++)
+      locsurf[i] = isidenticto[locsurf[i]];
+
+    for (int i = locsurf.Size()-1; i >= 0; i--)
+      {
+	bool indep = 1;
+	for (int j = 0; j < i; j++)
+	  if (locsurf[i] == locsurf[j])
 	    {
 	      indep = 0;
 	      break;
 	    }
-	}
-      if (!indep)
-	locsurf.Delete(i);
-    }
 
-  for (i = 0; i < locsurf.Size(); i++)
-    locsurf[i] = isidenticto[locsurf[i]];
-}
+	if (!indep) locsurf.Delete(i);
+      }
 
 
-void CSGeometry ::
-GetIndependentSurfaceIndices (const Solid * sol, 
-			      const Point<3> & p, Vec<3> & v,
-			      ARRAY<int> & locsurf) const
-{
-  Point<3> p2 = p + 1e-2 * v;
-  BoxSphere<3> box (p2, p2);
-  box.Increase (1e-3);
-  box.CalcDiamCenter();
-  GetIndependentSurfaceIndices (sol, box, locsurf);
-}
+    /*
+    // delete identified
+    for (int i = locsurf.Size()-1; i >= 0; i--)
+      {
+	bool indep = 1;
+	for (int j = 0; j < i; j++)
+	  {
+	    if (identicsurfaces.Used (INDEX_2::Sort (locsurf[i], locsurf[j])) !=
+		(isidenticto[locsurf[i]] == isidenticto[locsurf[j]]))
+	      {
+		cerr << "different result" << endl;
+		exit(1);
+	      }
 
+	    if (isidenticto[locsurf[i]] == isidenticto[locsurf[j]])
+	      {
+		indep = 0;
+		break;
+	      }
+	  }
+	if (!indep)
+	  locsurf.Delete(i);
+      }
 
-void CSGeometry :: 
-CalcTriangleApproximation(const Box<3> & boundingbox,
-			  double detail, double facets)
-{
-  PrintMessage (1, "Calc Triangle Approximation");
+    for (int i = 0; i < locsurf.Size(); i++)
+      locsurf[i] = isidenticto[locsurf[i]];
+    */
+  }
 
-  FindIdenticSurfaces (1e-6);
-  
-  int i, j, k;
-  //  int nms = GetNMainSolids ();
-  int ntlo = GetNTopLevelObjects();
 
-  triapprox.SetSize (ntlo);
-  ARRAY<int> surfind;
+  void CSGeometry ::
+  GetIndependentSurfaceIndices (const Solid * sol, 
+				const Point<3> & p, Vec<3> & v,
+				ARRAY<int> & locsurf) const
+  {
+    Point<3> p2 = p + 1e-2 * v;
+    BoxSphere<3> box (p2, p2);
+    box.Increase (1e-3);
+    box.CalcDiamCenter();
+    GetIndependentSurfaceIndices (sol, box, locsurf);
+  }
 
-  //  cout << "bb = " << boundingbox.PMin () << " - " << boundingbox.PMax() << endl;
 
-  for (i = 0; i < ntlo; i++)
-    {
-      Solid * sol;
-      Surface * surf;
-      GetTopLevelObject (i, sol, surf);
+  void CSGeometry :: 
+  CalcTriangleApproximation(const Box<3> & boundingbox,
+			    double detail, double facets)
+  {
+    PrintMessage (1, "Calc Triangle Approximation");
 
-      //      ((Solid*)MainSolid (i)) -> CalcSurfaceInverse ();
-      sol -> CalcSurfaceInverse ();
+    //    FindIdenticSurfaces (1e-6);
+  
+    int ntlo = GetNTopLevelObjects();
 
-      TriangleApproximation * tams = new TriangleApproximation();
-      triapprox[i] = tams;
+    for (int i = 0; i < triapprox.Size(); i++)
+      delete triapprox[i];
+    triapprox.SetSize (ntlo);
 
-      for (j = 0; j < GetNSurf(); j++)
-	{
-	  PrintMessageCR (3, "Surface ", j, "/", GetNSurf());
-	  if (surf && surf != GetSurface(j))
-	    continue;
+    ARRAY<int> surfind;
 
-	  TriangleApproximation tas;
-	  GetSurface (j) -> GetTriangleApproximation (tas, boundingbox, facets);
-	  
+    for (int i = 0; i < ntlo; i++)
+      {
+	Solid * sol;
+	Surface * surf;
+	GetTopLevelObject (i, sol, surf);
 
-	  int oldnp = tams -> GetNP();
+	sol -> CalcSurfaceInverse ();
 
-	  if (!tas.GetNP())
-	    continue;
+	TriangleApproximation * tams = new TriangleApproximation();
+	triapprox[i] = tams;
 
-	  for (k = 0; k < tas.GetNP(); k++)
-	    {
-	      Vec<3> n;
+	// sol -> GetSurfaceIndices (surfind);
+	for (int j = 0; j < GetNSurf(); j++)
+	  // for (int jj = 0; jj < surfind.Size(); jj++)
+	  {
+	    // int j = surfind[jj];
 
-	      tams -> AddPoint (tas.GetPoint(k));
-	      GetSurface(j) -> GetNormalVector (tas.GetPoint(k), n);
-	      n.Normalize();
-	      if (GetSurface(j)->Inverse()) n *= -1;
-	      tams -> AddNormal (n);
-	    }
+	    PrintMessageCR (3, "Surface ", j, "/", GetNSurf());
+	    // PrintMessageCR (3, "Surface ", j, "/", surfind.Size());
 
-	  
-	  BoxSphere<3> surfbox;
+	    if (surf && surf != GetSurface(j))
+	      continue;
 
-	  if (tas.GetNP())
-	    surfbox.Set (tas.GetPoint(0));
-	  for (k = 1; k < tas.GetNP(); k++)
-	    surfbox.Add (tas.GetPoint(k));
-	  surfbox.Increase (1e-6);
-	  surfbox.CalcDiamCenter();
+	    TriangleApproximation tas;
+	    GetSurface (j) -> GetTriangleApproximation (tas, boundingbox, facets);
 
-	  Solid * surflocsol = sol -> GetReducedSolid (surfbox);
-	  if (!surflocsol)
-	    continue;
+	    int oldnp = tams -> GetNP();
 
-	  for (k = 0; k < tas.GetNT(); k++)
-	    {
-	      const TATriangle & tri = tas.GetTriangle (k);
-
-	      // check triangle
-	      BoxSphere<3> box;
-	      box.Set (tas.GetPoint (tri[0]));
-	      box.Add (tas.GetPoint (tri[1]));
-	      box.Add (tas.GetPoint (tri[2]));
-	      box.Increase (1e-6);
-	      box.CalcDiamCenter();
-
-
-	      Solid * locsol = surflocsol -> GetReducedSolid (box);
-
-	      if (locsol)
-		{
-		  TATriangle tria(j, 
-				  tri[0] + oldnp,
-				  tri[1] + oldnp,
-				  tri[2] + oldnp);
-		  //		  tams->AddTriangle (tria);
-		  RefineTriangleApprox (locsol, j, box, detail, 
-					tria, *tams);
-		  delete locsol;
-		}
-	    }
+	    if (!tas.GetNP())
+	      continue;
 
-	}
+	    for (int k = 0; k < tas.GetNP(); k++)
+	      {
+		tams -> AddPoint (tas.GetPoint(k));
+		Vec<3> n = GetSurface(j) -> GetNormalVector (tas.GetPoint(k));
+		n.Normalize();
+		if (GetSurface(j)->Inverse()) n *= -1;
+		tams -> AddNormal (n);
+	      }
 
-      tams->RemoveUnusedPoints ();
-      PrintMessage (2, "Object ", i, " has ", tams->GetNT(), " triangles");
-    }
+	  
+	    BoxSphere<3> surfbox;
 
-  Change();
-}
+	    if (tas.GetNP())
+	      surfbox.Set (tas.GetPoint(0));
+	    for (int k = 1; k < tas.GetNP(); k++)
+	      surfbox.Add (tas.GetPoint(k));
+	    surfbox.Increase (1e-6);
+	    surfbox.CalcDiamCenter();
 
+	    Solid * surflocsol = sol -> GetReducedSolid (surfbox);
+	    if (!surflocsol)
+	      continue;
 
+	    for (int k = 0; k < tas.GetNT(); k++)
+	      {
+		const TATriangle & tri = tas.GetTriangle (k);
+
+		// check triangle
+		BoxSphere<3> box;
+		box.Set (tas.GetPoint (tri[0]));
+		box.Add (tas.GetPoint (tri[1]));
+		box.Add (tas.GetPoint (tri[2]));
+		box.Increase (1e-6);
+		box.CalcDiamCenter();
+
+
+		Solid * locsol = surflocsol -> GetReducedSolid (box);
+
+		if (locsol)
+		  {
+		    TATriangle tria(j, 
+				    tri[0] + oldnp,
+				    tri[1] + oldnp,
+				    tri[2] + oldnp);
+
+		    RefineTriangleApprox (locsol, j, box, detail, 
+					  tria, *tams);
+		    delete locsol;
+		  }
+	      }
+	  }
 
-void CSGeometry ::
-RefineTriangleApprox (Solid * locsol, 
-		      int surfind,
-		      const BoxSphere<3> & box, 
-		      double detail,
-		      const TATriangle & tria, 
-		      TriangleApproximation & tams)
-{
-  int i, j;
+	tams->RemoveUnusedPoints ();
+	PrintMessage (2, "Object ", i, " has ", tams->GetNT(), " triangles");
+      }
 
-  //  ARRAY<int> lsurfi;
-  int pinds[6];
+    Change();
+  }
 
-  static ARRAY<int> surfused;
-  surfused.SetSize(GetNSurf());
 
+
+  void CSGeometry ::
+  RefineTriangleApprox (Solid * locsol, 
+			int surfind,
+			const BoxSphere<3> & box, 
+			double detail,
+			const TATriangle & tria, 
+			TriangleApproximation & tams)
+  {
+    int pinds[6];
+    ArrayMem<int,500> surfused(GetNSurf());
   
-  ReducePrimitiveIterator rpi(box);
-  UnReducePrimitiveIterator urpi;
+    ReducePrimitiveIterator rpi(box);
+    UnReducePrimitiveIterator urpi;
 
 
-  locsol -> IterateSolid (rpi);
-  //  locsol -> GetSurfaceIndices (lsurfi);
+    locsol -> IterateSolid (rpi);
+    //  locsol -> GetSurfaceIndices (lsurfi);
 
 
-  IndexSet iset(GetNSurf());
-  locsol -> GetSurfaceIndices (iset);
-  const ARRAY<int> & lsurfi = iset.Array();
+    IndexSet iset(GetNSurf());
+    locsol -> GetSurfaceIndices (iset);
+    const ARRAY<int> & lsurfi = iset.Array();
 
-  locsol -> IterateSolid (urpi);
+    locsol -> IterateSolid (urpi);
 
 
-  int surfii = -1;
-  for (i = 0; i < lsurfi.Size(); i++)
-    if (lsurfi[i] == surfind)
-      {
-	surfii = i;
-	break;
-      }
+    int surfii = -1;
+    for (int i = 0; i < lsurfi.Size(); i++)
+      if (lsurfi[i] == surfind)
+	{
+	  surfii = i;
+	  break;
+	}
 
-  if (surfii == -1) 
-    return;
+    if (surfii == -1) 
+      return;
 
 
-  int cntindep = 0;
+    int cntindep = 0;
 
-  for (i = 0; i < lsurfi.Size(); i++)
-    {
-      int linkto = isidenticto[lsurfi[i]];
-      surfused[linkto] = 0;
-    }
+    for (int i = 0; i < lsurfi.Size(); i++)
+      {
+	int linkto = isidenticto[lsurfi[i]];
+	surfused[linkto] = 0;
+      }
 
-  for (i = 0; i < lsurfi.Size(); i++)
-    {
-      int linkto = isidenticto[lsurfi[i]];
-      if (!surfused[linkto])
-	{
-	  surfused[linkto] = 1;
-	  cntindep++;
-	}
-    }
+    for (int i = 0; i < lsurfi.Size(); i++)
+      {
+	int linkto = isidenticto[lsurfi[i]];
+	if (!surfused[linkto])
+	  {
+	    surfused[linkto] = 1;
+	    cntindep++;
+	  }
+      }
 
-  int inverse = surfaces[surfind]->Inverse();
+    int inverse = surfaces[surfind]->Inverse();
 
-  if (cntindep == 1)
-    {
-      //      (*testout) << "add trig of surf " << surfind << endl;
-      if (cntindep > 1) (*testout) << "not unique" << endl;
-      tams.AddTriangle (tria);
-      return;
-    }
+    if (cntindep == 1)
+      {
+	tams.AddTriangle (tria);
+	return;
+      }
 
-  if (cntindep == 2)
-    {
-      // just 2 surfaces:
-      // if smooth, project inner points to edge and finish
+    if (cntindep == 2)
+      {
+	// just 2 surfaces:
+	// if smooth, project inner points to edge and finish
 
-      int otherind;
+	int otherind = -1;
 
-      for (i = 0; i < lsurfi.Size(); i++)
-	{
-	  INDEX_2 i2 (lsurfi[i], surfind);
-	  i2.Sort();
+	for (int i = 0; i < lsurfi.Size(); i++)
+	  {
+	    INDEX_2 i2 (lsurfi[i], surfind);
+	    i2.Sort();
 	  
-	  if (i != surfii && !identicsurfaces.Used(i2))
-	    otherind = lsurfi[i];
-	}
-
-      double kappa = GetSurface(otherind)->
-	MaxCurvature (); // Loc (box.Center(), box.Diam());
-
-      if (kappa * box.Diam() < 0.1)
-	{
-	  int pnums[6];
-	  static int between[3][3] =
-	  { { 1, 2, 3 },
-	    { 0, 2, 4 },
-	    { 0, 1, 5 } };
-	  int onsurface[3];
-
-	  for (j = 0; j < 3; j++)
-	    {
-	      int pi = tria[j];
-	      pnums[j] = pi;
-	      onsurface[j] =  
-		!locsol->IsStrictIn (tams.GetPoint (pi), 1e-6) &&
-		locsol->IsIn (tams.GetPoint (pi), 1e-6);
-	    }
+	    if (i != surfii && !identicsurfaces.Used(i2))
+	      otherind = lsurfi[i];
+	  }
+
+	double kappa = GetSurface(otherind)-> MaxCurvature ();
+
+	if (kappa * box.Diam() < 0.1)
+	  {
+	    int pnums[6];
+	    static int between[3][3] =
+	      { { 1, 2, 3 },
+		{ 0, 2, 4 },
+		{ 0, 1, 5 } };
+	    int onsurface[3];
+
+	    for (int j = 0; j < 3; j++)
+	      {
+		int pi = tria[j];
+		pnums[j] = pi;
+		onsurface[j] =  
+		  !locsol->IsStrictIn (tams.GetPoint (pi), 1e-6) &&
+		  locsol->IsIn (tams.GetPoint (pi), 1e-6);
+	      }
 	  
-	  for (j = 0; j < 3; j++)
-	    {
-	      int lpi1 = between[j][0];
-	      int lpi2 = between[j][1];
-	      int lpin = between[j][2];
-	      if (onsurface[lpi1] == onsurface[lpi2])
-		pnums[lpin] = -1;
-	      else
-		{
-		  const Point<3> & p1 = tams.GetPoint (pnums[lpi1]);
-		  const Point<3> & p2 = tams.GetPoint (pnums[lpi2]);
-		  double f1 = GetSurface(otherind)->CalcFunctionValue (p1);
-		  double f2 = GetSurface(otherind)->CalcFunctionValue (p2);
-		  //		  Point<3> pn = Center (p1, p2);
-
-		  Point<3> pn;
-		  if ( fabs (f1-f2) > 1e-20 )
-		    {
-		      double l2 = -f1/(f2-f1);
-		      double l1 = f2/(f2-f1);
-		      pn = Point<3>(l1 * p1(0) + l2 * p2(0),
-				    l1 * p1(1) + l2 * p2(1),
-				    l1 * p1(2) + l2 * p2(2));
-		    }
-		  else
-		    {
+	    for (int j = 0; j < 3; j++)
+	      {
+		int lpi1 = between[j][0];
+		int lpi2 = between[j][1];
+		int lpin = between[j][2];
+		if (onsurface[lpi1] == onsurface[lpi2])
+		  pnums[lpin] = -1;
+		else
+		  {
+		    const Point<3> & p1 = tams.GetPoint (pnums[lpi1]);
+		    const Point<3> & p2 = tams.GetPoint (pnums[lpi2]);
+		    double f1 = GetSurface(otherind)->CalcFunctionValue (p1);
+		    double f2 = GetSurface(otherind)->CalcFunctionValue (p2);
+
+		    Point<3> pn;
+		    if ( fabs (f1-f2) > 1e-20 )
+		      {
+			double l2 = -f1/(f2-f1);
+			double l1 = f2/(f2-f1);
+			pn = Point<3>(l1 * p1(0) + l2 * p2(0),
+				      l1 * p1(1) + l2 * p2(1),
+				      l1 * p1(2) + l2 * p2(2));
+		      }
+		    else
 		      pn = p1;
-		    }
 
-		  pnums[lpin] = tams.AddPoint (pn);
+		    pnums[lpin] = tams.AddPoint (pn);
 
-		  GetSurface (surfind)->Project (pn);
+		    GetSurface (surfind)->Project (pn);
 		  
-		  Vec<3> n;
-		  GetSurface (surfind)->GetNormalVector (pn, n);
-		  if (inverse) n *= -1;
-		  tams.AddNormal(n);
-		}
-	    }
+		    Vec<3> n;
+		    n = GetSurface (surfind)->GetNormalVector (pn);
+		    if (inverse) n *= -1;
+		    tams.AddNormal(n);
+		  }
+	      }
 	  
-	  int vcase = 0;
-	  if (onsurface[0]) vcase++;
-	  if (onsurface[1]) vcase+=2;
-	  if (onsurface[2]) vcase+=4;
+	    int vcase = 0;
+	    if (onsurface[0]) vcase++;
+	    if (onsurface[1]) vcase+=2;
+	    if (onsurface[2]) vcase+=4;
 	  
-	  static int trias[8][6] =
-	  { { 0, 0, 0,   0, 0, 0 },
-	    { 1, 6, 5,   0, 0, 0 },
-	    { 2, 4, 6,   0, 0, 0 },
-	    { 1, 2, 4,   1, 4, 5 },
-	    { 3, 5, 4,   0, 0, 0 },
-	    { 1, 6, 4,   1, 4, 3 },
-	    { 2, 3, 6,   3, 5, 6 },
-	    { 1, 2, 3,   0, 0, 0 } };
-	  static int ntrias[4] =
-	  { 0, 1, 2, 1 };
-
-	  int nvis = 0;
-	  for (j = 0; j < 3; j++)
-	    if (onsurface[j])
-	      nvis++;
-
-	  for (j = 0; j < ntrias[nvis]; j++)
-	    {
-	      TATriangle ntria(tria.SurfaceIndex(),
-			       pnums[trias[vcase][3*j]-1],
-			       pnums[trias[vcase][3*j+1]-1],
-			       pnums[trias[vcase][3*j+2]-1]);
-	      tams.AddTriangle (ntria);
-	    }
+	    static int trias[8][6] =
+	      { { 0, 0, 0,   0, 0, 0 },
+		{ 1, 6, 5,   0, 0, 0 },
+		{ 2, 4, 6,   0, 0, 0 },
+		{ 1, 2, 4,   1, 4, 5 },
+		{ 3, 5, 4,   0, 0, 0 },
+		{ 1, 6, 4,   1, 4, 3 },
+		{ 2, 3, 6,   3, 5, 6 },
+		{ 1, 2, 3,   0, 0, 0 } };
+	    static int ntrias[4] =
+	      { 0, 1, 2, 1 };
+
+	    int nvis = 0;
+	    for (int j = 0; j < 3; j++)
+	      if (onsurface[j])
+		nvis++;
+
+	    for (int j = 0; j < ntrias[nvis]; j++)
+	      {
+		TATriangle ntria(tria.SurfaceIndex(),
+				 pnums[trias[vcase][3*j]-1],
+				 pnums[trias[vcase][3*j+1]-1],
+				 pnums[trias[vcase][3*j+2]-1]);
+		tams.AddTriangle (ntria);
+	      }
 
-	  /* saturn changes:
+	    /* saturn changes:
 
-	  int pvis[3];
-	  for (j = 0; j < 3; j++)
+	    int pvis[3];
+	    for (j = 0; j < 3; j++)
 	    pvis[j] = !locsol->IsStrictIn (tams.GetPoint (j+1), 1e-6) &&
-	      locsol->IsIn (tams.GetPoint (j+1), 1e-6);
+	    locsol->IsIn (tams.GetPoint (j+1), 1e-6);
 	  
-	  int newpi[3];
-	  for (j = 0; j < 3; j++)
+	    int newpi[3];
+	    for (j = 0; j < 3; j++)
+	    {
+	    int pi1 = j;
+	    int pi2 = (j+1) % 3;
+	    int pic = j;
+
+	    if (pvis[pi1] != pvis[pi2])
 	    {
-	      int pi1 = j;
-	      int pi2 = (j+1) % 3;
-	      int pic = j;
-
-	      if (pvis[pi1] != pvis[pi2])
-		{
-		  Point<3> hp = Center (tams.GetPoint (tria.PNum (pi1+1)),
-				       tams.GetPoint (tria.PNum (pi2+1)));
-
-		  newpi[j] = tams.AddPoint (hp);
-		  Vec<3> n = tams.GetNormal (pi1);
-		  tams.AddNormal (n);
-		}
-	      else
-		newpi[j] = 0;
+	    Point<3> hp = Center (tams.GetPoint (tria.PNum (pi1+1)),
+	    tams.GetPoint (tria.PNum (pi2+1)));
+
+	    newpi[j] = tams.AddPoint (hp);
+	    Vec<3> n = tams.GetNormal (pi1);
+	    tams.AddNormal (n);
+	    }
+	    else
+	    newpi[j] = 0;
 	    }
 
-	  int nvis = 0;
-	  for (j = 0; j <= nvis; j++)
+	    int nvis = 0;
+	    for (j = 0; j <= nvis; j++)
 	    if (pvis[j]) nvis++;
 
-	  int si = tria.SurfaceIndex();
-	  switch (nvis)
+	    int si = tria.SurfaceIndex();
+	    switch (nvis)
 	    {
 	    case 0:
-	      break;
+	    break;
 	    case 1:
-	      {
-		int visj;
-		for (j = 0; j < 3; j++)
-		  if (pvis[j]) visj = j;
-		int pivis = tria.PNum (visj+1);
-		int pic1 = newpi[(visj+1)%3];
-		int pic2 = newpi[(visj+2)%3];
+	    {
+	    int visj;
+	    for (j = 0; j < 3; j++)
+	    if (pvis[j]) visj = j;
+	    int pivis = tria.PNum (visj+1);
+	    int pic1 = newpi[(visj+1)%3];
+	    int pic2 = newpi[(visj+2)%3];
 		
-		cout << pivis << "," << pic1 << "," << pic2 << endl;
+	    cout << pivis << "," << pic1 << "," << pic2 << endl;
 		
-		tams.AddTriangle (TATriangle (si, pivis, pic1,pic2));
-		break;
-	      }
+	    tams.AddTriangle (TATriangle (si, pivis, pic1,pic2));
+	    break;
+	    }
 	    case 2:
-	      {
-		int nvisj;
-		for (j = 0; j < 3; j++)
-		  if (!pvis[j]) nvisj = j;
-
-		int pivis1 = tria.PNum ((nvisj+1)%3+1);
-		int pivis2 = tria.PNum ((nvisj+2)%3+1);
-		int pic1 = newpi[nvisj];
-		int pic2 = newpi[(nvisj+2)%3];
-
-		tams.AddTriangle (TATriangle (si, pivis1, pic1,pic2));
-		tams.AddTriangle (TATriangle (si, pivis1, pic1,pivis2));
-		break;
-	      }
+	    {
+	    int nvisj;
+	    for (j = 0; j < 3; j++)
+	    if (!pvis[j]) nvisj = j;
+
+	    int pivis1 = tria.PNum ((nvisj+1)%3+1);
+	    int pivis2 = tria.PNum ((nvisj+2)%3+1);
+	    int pic1 = newpi[nvisj];
+	    int pic2 = newpi[(nvisj+2)%3];
+
+	    tams.AddTriangle (TATriangle (si, pivis1, pic1,pic2));
+	    tams.AddTriangle (TATriangle (si, pivis1, pic1,pivis2));
+	    break;
+	    }
 	    case 3:
-	      {
-		tams.AddTriangle (tria);
-		break;
-	      }
+	    {
+	    tams.AddTriangle (tria);
+	    break;
+	    }
 	    }
 
-	  */
-	  return;
-	}
-    }
+	    */
+	    return;
+	  }
+      }
 
-  // bisection
-  if (box.Diam() < detail)
-    return;
-  
-  for (i = 0; i < 3; i++)
-    pinds[i] = tria[i];
+    // bisection
+    if (box.Diam() < detail)
+      return;
+    
+    for (int i = 0; i < 3; i++)
+      pinds[i] = tria[i];
   
-  static int between[3][3] =
-  { { 0, 1, 5 },
-    { 0, 2, 4 },
-    { 1, 2, 3 } };
+    static int between[3][3] =
+      { { 0, 1, 5 },
+	{ 0, 2, 4 },
+	{ 1, 2, 3 } };
   
-  for (i = 0; i < 3; i++)
-    {
-      int pi1 = tria[between[i][0]];
+    for (int i = 0; i < 3; i++)
+      {
+	// int pi1 = tria[between[i][0]];
 
-      Point<3> newp = Center (tams.GetPoint (tria[between[i][0]]),
-			      tams.GetPoint (tria[between[i][1]]));
-      Vec<3> n;
+	Point<3> newp = Center (tams.GetPoint (tria[between[i][0]]),
+				tams.GetPoint (tria[between[i][1]]));
+	Vec<3> n;
       
-      GetSurface(surfind)->Project (newp);
-      GetSurface(surfind)->GetNormalVector (newp, n);
+	GetSurface(surfind)->Project (newp);
+	n = GetSurface(surfind)->GetNormalVector (newp);
       
-      pinds[between[i][2]] = tams.AddPoint (newp);
-      if (inverse) n *= -1;
-      tams.AddNormal (n);
-    }
+	pinds[between[i][2]] = tams.AddPoint (newp);
+	if (inverse) n *= -1;
+	tams.AddNormal (n);
+      }
   
-  static int trias[4][4] =
-  { { 0, 5, 4 },
-    { 5, 1, 3 },
-    { 4, 3, 2 },
-    { 3, 4, 5 } };
+    static int trias[4][4] =
+      { { 0, 5, 4 },
+	{ 5, 1, 3 },
+	{ 4, 3, 2 },
+	{ 3, 4, 5 } };
   
-  for (i = 0; i < 4; i++)
-    {
-      TATriangle ntri(surfind,
-		      pinds[trias[i][0]],
-		      pinds[trias[i][1]],
-		      pinds[trias[i][2]]);
-
-      // check triangle
-      BoxSphere<3> nbox;
-      nbox.Set (tams.GetPoint (ntri[0]));
-      nbox.Add (tams.GetPoint (ntri[1]));
-      nbox.Add (tams.GetPoint (ntri[2]));
-      nbox.Increase (1e-6);
-      nbox.CalcDiamCenter();
-
-      Solid * nsol = locsol -> GetReducedSolid (nbox);
-
-      if (nsol)
-	{
-	  RefineTriangleApprox (nsol, surfind, nbox, 
-				detail, ntri, tams);
+    for (int i = 0; i < 4; i++)
+      {
+	TATriangle ntri(surfind,
+			pinds[trias[i][0]],
+			pinds[trias[i][1]],
+			pinds[trias[i][2]]);
+
+	// check triangle
+	BoxSphere<3> nbox;
+	nbox.Set (tams.GetPoint (ntri[0]));
+	nbox.Add (tams.GetPoint (ntri[1]));
+	nbox.Add (tams.GetPoint (ntri[2]));
+	nbox.Increase (1e-6);
+	nbox.CalcDiamCenter();
+
+	Solid * nsol = locsol -> GetReducedSolid (nbox);
+
+	if (nsol)
+	  {
+	    RefineTriangleApprox (nsol, surfind, nbox, 
+				  detail, ntri, tams);
 	  
-	  delete nsol;
-	}
-    }
-}
-
-
-
-
-
-class ClearVisitedIt : public SolidIterator
-{
-public:
-  ClearVisitedIt ();
-  ~ClearVisitedIt ();
-  virtual void Do (Solid * sol);
-};
-
-ClearVisitedIt :: ClearVisitedIt ()
-{
-  ;
-}
-ClearVisitedIt :: ~ClearVisitedIt ()
-{
-  ;
-}
+	    delete nsol;
+	  }
+      }
+  }
 
-void ClearVisitedIt :: Do (Solid * sol)
-{
-  sol -> visited = 0;
-}
 
 
 
-void CSGeometry :: 
-IterateAllSolids (SolidIterator & it, int only_once)
-{
-  int i;
+  class ClearVisitedIt : public SolidIterator
+  {
+  public:
+    ClearVisitedIt () { ; }
+    virtual ~ClearVisitedIt () { ; }
 
-  if (only_once)
-    {
-      ClearVisitedIt clit;
-      for (i = 0; i < solids.Size(); i++)
-	solids[i] -> IterateSolid (clit, 0);
-      
+    virtual void Do (Solid * sol)
+    { 
+      sol -> visited = 0;
     }
-
-  for (i = 0; i < solids.Size(); i++)
-    solids[i] -> IterateSolid (it, only_once);
-}
-
+  };
 
 
-/*
-const Box<3> & CSGeometry :: BoundingBox () const
-{
-  return boundingbox;
-}
-*/
-
-void CSGeometry :: SetBoundingBox (const Box<3> & abox)
-{
-  boundingbox = abox;
-}
+  void CSGeometry :: 
+  IterateAllSolids (SolidIterator & it, int only_once)
+  {
+    if (only_once)
+      {
+	ClearVisitedIt clit;
+	for (int i = 0; i < solids.Size(); i++)
+	  solids[i] -> IterateSolid (clit, 0);
+      }
 
-double CSGeometry ::  MaxSize () const
-{
-  double maxs, mins;
-  maxs = max3 (boundingbox.PMax()(0), 
-	       boundingbox.PMax()(1), 
-	       boundingbox.PMax()(2));
-  mins = min3 (boundingbox.PMin()(0), 
-	       boundingbox.PMin()(1), 
-	       boundingbox.PMin()(2));
-  return max2 (maxs, -mins) * 1.1;
-}
+    for (int i = 0; i < solids.Size(); i++)
+      solids[i] -> IterateSolid (it, only_once);
+  }
+
+
+  double CSGeometry ::  MaxSize () const
+  {
+    double maxs, mins;
+    maxs = max3 (boundingbox.PMax()(0), 
+		 boundingbox.PMax()(1), 
+		 boundingbox.PMax()(2));
+    mins = min3 (boundingbox.PMin()(0), 
+		 boundingbox.PMin()(1), 
+		 boundingbox.PMin()(2));
+    return max2 (maxs, -mins) * 1.1;
+  }
 }
diff --git a/Netgen/libsrc/csg/csgeom.hpp b/Netgen/libsrc/csg/csgeom.hpp
index 4a8e32b28f..6e29a8163b 100644
--- a/Netgen/libsrc/csg/csgeom.hpp
+++ b/Netgen/libsrc/csg/csgeom.hpp
@@ -104,6 +104,9 @@ private:
   /// bounding box of geometry
   Box<3> boundingbox;
 
+  /// bounding box, if not set by input file
+  static Box<3> default_boundingbox;
+
   /// identic surfaces are stored by pair of indizes, val = inverse
   INDEX_2_HASHTABLE<int> identicsurfaces;
   ARRAY<int> isidenticto;
@@ -130,10 +133,12 @@ public:
 
   void AddSurface (Surface * surf);
   void AddSurface (char * name, Surface * surf);
+  void AddSurfaces (Primitive * prim);
 
   int GetNSurf () const { return surfaces.Size(); }
   const Surface * GetSurface (const char * name) const;
-  const Surface * GetSurface (int i) const;
+  const Surface * GetSurface (int i) const
+  { return surfaces[i]; }
 
   void SetSolid (const char * name, Solid * sol);
   const Solid * GetSolid (const char * name) const;
@@ -145,22 +150,21 @@ public:
 
   void SetFlags (const char * solidname, const Flags & flags);
 
-  /*
-  ///
-  void FindMainSolids ();
-  ///
-  int GetNMainSolids () const
-    { return mainsolids.Size(); }
-  ///
-  const Solid* MainSolid(int i) const
-    { return mainsolids.Get(i); }
-  */
-
 
-  int GetNTopLevelObjects () const { return toplevelobjects.Size(); }
+  int GetNTopLevelObjects () const
+  { return toplevelobjects.Size(); }
   int SetTopLevelObject (Solid * sol, Surface * surf = NULL);
-  void GetTopLevelObject (int nr, Solid *& sol, Surface *& surf);
-  void GetTopLevelObject (int nr, const Solid *& sol, const Surface *& surf) const;
+  void GetTopLevelObject (int nr, Solid *& sol, Surface *& surf)
+  {
+    sol = toplevelobjects[nr]->GetSolid();
+    surf = toplevelobjects[nr]->GetSurface();
+  }
+  void GetTopLevelObject (int nr, const Solid *& sol, const Surface *& surf) const
+  {
+    sol = toplevelobjects[nr]->GetSolid();
+    surf = toplevelobjects[nr]->GetSurface();
+  }
+
   TopLevelObject * GetTopLevelObject (const Solid * sol, const Surface * surf = NULL);
   TopLevelObject * GetTopLevelObject (int nr)
   { return toplevelobjects[nr]; }
@@ -178,6 +182,7 @@ public:
   
 
   // quick implementations:
+  ARRAY<SingularFace*> singfaces;
   ARRAY<SingularEdge*> singedges;
   ARRAY<SingularPoint*> singpoints;
   ARRAY<Identification*> identifications;
@@ -203,6 +208,7 @@ public:
   ///
   int GetSurfaceClassRepresentant (int si) const
     { return isidenticto[si]; }
+
   ///
   const TriangleApproximation * GetTriApprox (int msnr)
   {
@@ -222,7 +228,17 @@ public:
 			     TriangleApproximation & tams);
 
   const Box<3> & BoundingBox () const { return boundingbox; }
-  void SetBoundingBox (const Box<3> & abox);
+
+  void SetBoundingBox (const Box<3> & abox)
+  {
+    boundingbox = abox;
+  }
+
+
+  static void SetDefaultBoundingBox (const Box<3> & abox)
+  {
+    default_boundingbox = abox;
+  }
 
   double MaxSize () const;
 
diff --git a/Netgen/libsrc/csg/csgparser.cpp b/Netgen/libsrc/csg/csgparser.cpp
index 9372d5675d..b6d4878399 100644
--- a/Netgen/libsrc/csg/csgparser.cpp
+++ b/Netgen/libsrc/csg/csgparser.cpp
@@ -16,7 +16,7 @@ namespace netgen
       TOK_EQU = '=', TOK_COMMA = ',', TOK_SEMICOLON = ';',
       TOK_NUM = 100, TOK_STRING, TOK_NAMED_SOLID, TOK_PRIMITIVE, 
       TOK_OR, TOK_AND, TOK_NOT, 
-      TOK_SINGULAR, TOK_EDGE, TOK_POINT, TOK_IDENTIFY, TOK_CLOSESURFACES,
+      TOK_SINGULAR, TOK_EDGE, TOK_POINT, TOK_FACE, TOK_IDENTIFY, TOK_CLOSESURFACES,
       TOK_CLOSEEDGES, TOK_PERIODIC,
       TOK_SOLID, TOK_RECO, TOK_TLO, TOK_BOUNDINGBOX, TOK_BOUNDARYCONDITION,
       TOK_END };
@@ -38,6 +38,7 @@ namespace netgen
       { TOK_NOT,     "not" },
       { TOK_SINGULAR, "singular" },
       { TOK_EDGE,     "edge" },
+      { TOK_FACE,     "face" },
       { TOK_POINT,    "point" },
       { TOK_IDENTIFY, "identify" },
       { TOK_CLOSESURFACES, "closesurfaces" },
@@ -48,10 +49,13 @@ namespace netgen
     };
 
   enum PRIMITIVE_TYPE
-    { TOK_SPHERE = 1, TOK_CYLINDER, TOK_PLANE, TOK_ELLIPTICCYLINDER, 
-      TOK_ELLIPSOID,
-      TOK_CONE, TOK_TUBE,
-      TOK_GENCYL, TOK_ORTHOBRICK, TOK_POLYHEDRON, TOK_EXTRUSION, TOK_REVOLUTION,
+    {
+      TOK_SPHERE = 1, TOK_CYLINDER, TOK_PLANE, TOK_ELLIPTICCYLINDER, 
+      TOK_ELLIPSOID, TOK_CONE, 
+      TOK_ORTHOBRICK, TOK_POLYHEDRON, 
+      
+      TOK_TUBE, TOK_GENCYL, TOK_EXTRUSION, TOK_REVOLUTION,    // currently, out of order
+
       TOK_TRANSLATE, TOK_MULTITRANSLATE, TOK_ROTATE, TOK_MULTIROTATE
     };
 
@@ -69,10 +73,11 @@ namespace netgen
       { TOK_CONE,      "cone" },
       { TOK_ELLIPTICCYLINDER, "ellipticcylinder" },
       { TOK_ELLIPSOID, "ellipsoid" },
-      { TOK_TUBE,      "tube" },
-      { TOK_GENCYL,    "gencyl" },
       { TOK_ORTHOBRICK, "orthobrick" },
       { TOK_POLYHEDRON, "polyhedron" },
+
+      { TOK_TUBE,      "tube" },
+      { TOK_GENCYL,    "gencyl" },
       { TOK_EXTRUSION,  "extrusion" },
       { TOK_REVOLUTION, "revolution" },
 
@@ -83,24 +88,8 @@ namespace netgen
       { PRIMITIVE_TYPE(0) }
     };
 
-
-
   static CSGeometry * geom;
 
-  /*
-%token <solidtype> TOK_SPHERE TOK_CYLINDER TOK_CONE TOK_PLAIN TOK_TUBE TOK_GENCYL TOK_ORTHOBRICK TOK_POLYHEDRON TOK_REVOLUTION
-%left <solidtype> TOK_OR TOK_AND TOK_NOT
-%token <solidtype> TOK_TRANSLATE TOK_MULTITRANSLATE TOK_ROTATE TOK_MULTIROTATE
-%type <solidtype> solid solidprimitive 
-%type <void> splinesegmentlist splinesegment readbspline bsplinepointlist
-%type <chptr> anyident
-%token TOK_SINGULAR TOK_EDGE TOK_POINT
-%token TOK_IDENTIFY TOK_CLOSESURFACES TOK_CLOSEEDGES TOK_PERIODIC
-%token TOK_BOUNDARYCONDITION
-%type <void> polyhedronpoints polyhedronfaces polyhedronpoint polyhedronface
-%type <void> revolutionpoints revolutionpoint
-  */
-
 
   
   class CSGScanner
@@ -133,6 +122,14 @@ namespace netgen
     { return prim_token; }
   
     void ReadNext();
+
+    /*
+    CSGScanner & Parse (char ch);
+    CSGScanner & Parse (int & i);
+    CSGScanner & Parse (double & d);
+    CSGScanner & Parse (Point<3> & p);
+    CSGScanner & Parse (Vec<3> & p);
+    */
     void Error (const string & err);
   };
 
@@ -151,7 +148,7 @@ namespace netgen
     char ch;
   
 
-    // whitespaces ueberspringen
+    // scan whitespaces
     do
       { 
 	scanin->get(ch);
@@ -208,17 +205,14 @@ namespace netgen
 	    {
 	      string_value = string (1, ch);
 	      scanin->get(ch);
-	      while (isalnum(ch))
+	      while (isalnum(ch) || ch == '_')
 		{
 		  string_value += ch;
 		  scanin->get(ch);
 		}
 	      scanin->putback (ch);
 	    }
-	  /*
-	  (*scanin).putback (ch);
-	  (*scanin) >> string_value;
-	  */
+
 	  int nr = 0;
 	  while (defkw[nr].kw)
 	    {
@@ -259,15 +253,12 @@ namespace netgen
     Solid = Term { OR Term }
     Term  = Primary { AND Primary }
     Primary = PRIM | IDENT | ( Solid ) | NOT Primary
-   */
+  */
 
   void ParseChar (CSGScanner & scan, char ch)
   {
-    char str[2];
-    str[0] = ch;
-    str[1] = 0;
     if (scan.GetToken() != TOKEN_TYPE(ch)) 
-      scan.Error (string ("token '") + string(str) + string("' expected"));
+      scan.Error (string ("token '") + string(1, ch) + string("' expected"));
     scan.ReadNext();
   }
   
@@ -284,6 +275,50 @@ namespace netgen
     return val;
   }
 
+  Vec<3> ParseVector (CSGScanner & scan)
+  {
+    Vec<3> v;
+    v(0) = ParseNumber (scan);
+    ParseChar (scan, ',');
+    v(1) = ParseNumber (scan);
+    ParseChar (scan, ',');
+    v(2) = ParseNumber (scan);
+    return v;
+  }
+
+
+  CSGScanner & operator>> (CSGScanner & scan, char ch)
+  {
+    if (scan.GetToken() != TOKEN_TYPE(ch)) 
+      scan.Error (string ("token '") + string(1, ch) + string("' expected"));
+    scan.ReadNext();
+    return scan;
+  }
+
+  CSGScanner & operator>> (CSGScanner & scan, double & d)
+  {
+    d = ParseNumber (scan);
+    return scan;
+  }
+
+  CSGScanner & operator>> (CSGScanner & scan, int & i)
+  {
+    i = int (ParseNumber (scan));
+    return scan;
+  }
+
+  CSGScanner & operator>> (CSGScanner & scan, Point<3> & p)
+  {
+    scan >> p(0) >> ',' >> p(1) >> ',' >> p(2);
+    return scan;
+  }
+
+  CSGScanner & operator>> (CSGScanner & scan, Vec<3> & v)
+  {
+    scan >> v(0) >> ',' >> v(1) >> ',' >> v(2);
+    return scan;
+  }
+
 
   Solid * ParseSolid (CSGScanner & scan);
   Solid * ParseTerm (CSGScanner & scan);
@@ -294,7 +329,6 @@ namespace netgen
   {
     if (scan.GetToken() == TOK_PRIMITIVE)
       {
-	//	cout << "prim token = " << int (scan.GetPrimitiveToken()) << endl;
 	switch (scan.GetPrimitiveToken())
 	  {
 	  case TOK_PLANE:
@@ -303,57 +337,23 @@ namespace netgen
 	      Vec<3> v;
 	      
 	      scan.ReadNext();
-	      
-	      ParseChar (scan, '(');
-	      p(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      p(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      p(2) = ParseNumber (scan);
-	      ParseChar (scan, ';');
-	      v(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      v(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      v(2) = ParseNumber (scan);
-	      ParseChar (scan, ')');
+	      scan >> '(' >> p >> ';' >> v >> ')';
 
-	      // cout << "define plane, p = " << p << "; v = " << v << endl;
 	      OneSurfacePrimitive * surf = new Plane ( p, v );
-
-	      geom->AddSurface (surf);
-	      surf->SetSurfaceId (0, geom->GetNSurf()-1);
-
+	      geom->AddSurfaces (surf);
 	      return new Solid (surf);
 	    }
+
 	  case TOK_CYLINDER:
 	    {
 	      Point<3> pa, pb;
 	      double r;
 	      
 	      scan.ReadNext();
-	      
-	      ParseChar (scan, '(');
-	      pa(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pa(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pa(2) = ParseNumber (scan);
-	      ParseChar (scan, ';');
-	      pb(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pb(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pb(2) = ParseNumber (scan);
-	      ParseChar (scan, ';');
-	      r = ParseNumber (scan);
-	      ParseChar (scan, ')');
-	      
-	      OneSurfacePrimitive * surf = new Cylinder ( pa, pb, r );
-
-	      geom->AddSurface (surf);
-	      surf->SetSurfaceId (0, geom->GetNSurf()-1);
+	      scan >> '(' >> pa >> ';' >> pb >> ';' >> r >> ')';
 
+	      OneSurfacePrimitive * surf = new Cylinder ( pa, pb, r );
+	      geom->AddSurfaces (surf);
 	      return new Solid (surf);
 	    }
 
@@ -363,34 +363,10 @@ namespace netgen
 	      Vec<3> vl, vs;
 	      
 	      scan.ReadNext();
-	      
-	      ParseChar (scan, '(');
-	      pa(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pa(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pa(2) = ParseNumber (scan);
-	      ParseChar (scan, ';');
-
-	      vl(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      vl(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      vl(2) = ParseNumber (scan);
-	      ParseChar (scan, ';');
+	      scan >> '(' >> pa >> ';' >> vl >> ';' >> vs >> ')';
 
-	      vs(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      vs(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      vs(2) = ParseNumber (scan);
-	      ParseChar (scan, ')');
-	      
 	      OneSurfacePrimitive * surf = new EllipticCylinder ( pa, vl, vs);
-
-	      geom->AddSurface (surf);
-	      surf->SetSurfaceId (0, geom->GetNSurf()-1);
-
+	      geom->AddSurfaces (surf);
 	      return new Solid (surf);
 	    }
 
@@ -401,41 +377,10 @@ namespace netgen
 	      Vec<3> v1, v2, v3;
 	      
 	      scan.ReadNext();
-	      
-	      ParseChar (scan, '(');
-	      pa(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pa(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pa(2) = ParseNumber (scan);
-	      ParseChar (scan, ';');
-
-	      v1(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      v1(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      v1(2) = ParseNumber (scan);
-	      ParseChar (scan, ';');
-
-	      v2(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      v2(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      v2(2) = ParseNumber (scan);
-	      ParseChar (scan, ';');
+	      scan >> '(' >> pa >> ';' >> v1 >> ';' >> v2 >> ';' >> v3 >> ')';
 
-	      v3(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      v3(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      v3(2) = ParseNumber (scan);
-	      ParseChar (scan, ')');
-	      
 	      OneSurfacePrimitive * surf = new Ellipsoid ( pa, v1, v2, v3);
-
-	      geom->AddSurface (surf);
-	      surf->SetSurfaceId (0, geom->GetNSurf()-1);
-
+	      geom->AddSurfaces (surf);
 	      return new Solid (surf);
 	    }
 
@@ -446,59 +391,25 @@ namespace netgen
 	      double ra, rb;
 	      
 	      scan.ReadNext();
-	      
-	      ParseChar (scan, '(');
-	      pa(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pa(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pa(2) = ParseNumber (scan);
-	      ParseChar (scan, ';');
-	      ra = ParseNumber (scan);
-	      ParseChar (scan, ';');
-	      pb(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pb(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pb(2) = ParseNumber (scan);
-	      ParseChar (scan, ';');
-	      rb = ParseNumber (scan);
-	      ParseChar (scan, ')');
-	      
-	      OneSurfacePrimitive * surf = new Cone ( pa, pb, ra, rb);
-
-	      geom->AddSurface (surf);
-	      surf->SetSurfaceId (0, geom->GetNSurf()-1);
+	      scan >> '(' >> pa >> ';' >> ra >> ';' >> pb >> ';' >> rb >> ')';
 
+	      OneSurfacePrimitive * surf = new Cone ( pa, pb, ra, rb );
+	      geom->AddSurfaces (surf);
 	      return new Solid (surf);
 	    }
 
 
 
-
 	  case TOK_SPHERE:
 	    {
 	      Point<3> p;
 	      double r;
 	      
 	      scan.ReadNext();
-	      
-	      ParseChar (scan, '(');
-	      p(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      p(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      p(2) = ParseNumber (scan);
-	      ParseChar (scan, ';');
-	      r = ParseNumber (scan);
-	      ParseChar (scan, ')');
-	      
-	      // cout << "define sphere, c = " << p << ", rad = " << r << endl;
-	      OneSurfacePrimitive * surf = new Sphere ( p, r );
-
-	      geom->AddSurface (surf);
-	      surf->SetSurfaceId (0, geom->GetNSurf()-1);
+	      scan >> '(' >> p >> ';' >> r >> ')';
 
+	      OneSurfacePrimitive * surf = new Sphere ( p, r );
+	      geom->AddSurfaces (surf);
 	      return new Solid (surf);
 	    }
 
@@ -507,36 +418,13 @@ namespace netgen
 	      Point<3> pa, pb;
 	      
 	      scan.ReadNext();
+	      scan >> '(' >> pa >> ';' >> pb >> ')';
 	      
-	      ParseChar (scan, '(');
-	      pa(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pa(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pa(2) = ParseNumber (scan);
-	      ParseChar (scan, ';');
-	      pb(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pb(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      pb(2) = ParseNumber (scan);
-	      ParseChar (scan, ')');
-	      
-	      // cout << "define orthobrick, p1 = " << pa << "; p2 = " << pb << endl;
-
 	      Primitive * nprim = new OrthoBrick (pa, pb);
-	      
-	      for (int j = 0; j < nprim->GetNSurfaces(); j++)
-		{
-		  geom->AddSurface (&nprim->GetSurface(j));
-		  nprim->SetSurfaceId (j, geom->GetNSurf()-1);
-		}
+	      geom->AddSurfaces (nprim);
 	      return new Solid (nprim);
 	    } 
 
-
-	    
-
 	  case TOK_POLYHEDRON:
 	    {
 	      // Added by Dalibor Lukas, October 15, 2003
@@ -552,15 +440,9 @@ namespace netgen
 	      // scanning the points
 	      while (1)
 		{
-		  p(0) = ParseNumber (scan);
-		  ParseChar (scan, ',');
-		  p(1) = ParseNumber (scan);
-		  ParseChar (scan, ',');
-		  p(2) = ParseNumber (scan);
+		  p = Point<3> (ParseVector (scan));
 		  ParseChar (scan, ';');
 
-		  // cout << "point = " << p << endl;
-
 		  polyhedron->AddPoint(p);
 
 		  if (scan.GetToken() == ';')
@@ -595,19 +477,14 @@ namespace netgen
 		    }
 		  scan.ReadNext();
 		}
-	      
-	      for (int j = 0; j < polyhedron->GetNSurfaces(); j++)
-		{
-		  geom->AddSurface (&polyhedron->GetSurface(j));
-		  polyhedron->SetSurfaceId (j, geom->GetNSurf()-1);
-		}
 
+	      geom->AddSurfaces (polyhedron);
 	      return new Solid (polyhedron);
 	    }
 
 
-	  case TOK_EXTRUSION:
-	    {
+	  case TOK_EXTRUSION:     // not functional
+	    {   
 	      Point<3> p0;
 	      Vec<3> ex, ey;
 	      ARRAY<Point<2> > points;
@@ -638,8 +515,8 @@ namespace netgen
 
 	      cout << "p0 = " << p0 << endl;
 
-	      int npseg = 0;
-	      int nseg = 0;
+	      // int npseg = 0;
+	      // int nseg = 0;
 	      while (1)
 		{
 		  Point<2> p1, p2, p3;
@@ -658,45 +535,45 @@ namespace netgen
 
 
 	      /*
-	      while (1)
+		while (1)
 		{
-		  Point<2> p1, p2, p3;
+		Point<2> p1, p2, p3;
 		  
-		  p3 = p2;
-		  p2 = p1;
-		  p1(0) = ParseNumber(scan);
-		  ParseChar (scan, ',');
-		  p1(1) = ParseNumber(scan);
-		  npseg++;
+		p3 = p2;
+		p2 = p1;
+		p1(0) = ParseNumber(scan);
+		ParseChar (scan, ',');
+		p1(1) = ParseNumber(scan);
+		npseg++;
 		  
-		  cout << "p1 = " << p1 << endl;
+		cout << "p1 = " << p1 << endl;
 
-		  if (scan.GetToken() == ';' || scan.GetToken() == ')')
-		    {
-		      if (npseg == 2)
-			{
-			  p3 = p2;
-			  p2 = Center (p1, p3);
-			}
-		      if (nseg == 0)
-			points.Append (p3);
-		      points.Append (p2);
-		      points.Append (p1);
-		      npseg = 1;
-		      nseg++;
-
-		      cout << "p1, = " << p1 << ", p2 = " << p2 << ", p3 = " << p3 << endl;
-		    }
+		if (scan.GetToken() == ';' || scan.GetToken() == ')')
+		{
+		if (npseg == 2)
+		{
+		p3 = p2;
+		p2 = Center (p1, p3);
+		}
+		if (nseg == 0)
+		points.Append (p3);
+		points.Append (p2);
+		points.Append (p1);
+		npseg = 1;
+		nseg++;
+
+		cout << "p1, = " << p1 << ", p2 = " << p2 << ", p3 = " << p3 << endl;
+		}
 		  
-		  if (scan.GetToken() == ')')
-		    {
-		      scan.ReadNext();
-		      break;
-		    }
-		  if (scan.GetToken() == ';' || scan.GetToken() == ',')
-		    {
-		      scan.ReadNext();
-		    }
+		if (scan.GetToken() == ')')
+		{
+		scan.ReadNext();
+		break;
+		}
+		if (scan.GetToken() == ';' || scan.GetToken() == ',')
+		{
+		scan.ReadNext();
+		}
 		}
 	      */
 	      cout << "p0 = " << p0 << endl;
@@ -705,13 +582,9 @@ namespace netgen
 	      
 	      Extrusion * extrusion = new Extrusion (p0, ex, ey, points);
 	      
-	      for (int i = 0; i < extrusion->GetNSurfaces(); i++)
-		{
-		  geom->AddSurface (&extrusion->GetSurface(i));
-		  extrusion->SetSurfaceId(i, geom->GetNSurf()-1);
-		}
+	      geom->AddSurfaces (extrusion);
 	      return new Solid (extrusion);
-		    
+	      
 	      /*
 	      // cout << "define cylinder, pa = " << pa << "; pb = " << pb
 	      // << ", rad = " << r << endl;
@@ -732,12 +605,9 @@ namespace netgen
 	    {
 	      Vec<3> v;
 	      scan.ReadNext();
+
 	      ParseChar (scan, '(');
-	      v(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      v(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      v(2) = ParseNumber (scan);
+	      v = ParseVector (scan);
 	      ParseChar (scan, ';');
 	      
 	      Solid * sol1 = ParseSolid (scan);
@@ -757,24 +627,15 @@ namespace netgen
 	      int n;
 	      
 	      scan.ReadNext();
-	      ParseChar (scan, '(');
-	      v(0) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      v(1) = ParseNumber (scan);
-	      ParseChar (scan, ',');
-	      v(2) = ParseNumber (scan);
-	      ParseChar (scan, ';');
-	      
-	      n = int (ParseNumber (scan));
-	      ParseChar (scan, ';');
-	      
+
+	      scan >> '(' >> v >> ';' >> n >> ';';
+
 	      Solid * sol1 = ParseSolid (scan);
 	      
-	      ParseChar (scan, ')');
+	      scan >> ')';
 	      
-	      int i;
 	      Solid * hsol = sol1;
-	      for (i = 1; i <= n; i++)
+	      for (int i = 1; i <= n; i++)
 		{
 		  Solid * nsol = sol1 -> Copy(*geom);
 		  Transformation<3> trans(double(i) * v);
@@ -786,6 +647,37 @@ namespace netgen
 	    }
 
 
+	  case TOK_MULTIROTATE: 
+	    {
+	      Point<3> c;
+	      Vec<3> v;
+	      int n;
+	      
+	      scan.ReadNext();
+
+	      scan >> '(' >> c >> ';' >> v >> ';' >> n >> ';';
+	      Solid * sol1 = ParseSolid (scan);
+	      scan >> ')';
+
+	      Transformation<3> trans(c, v(0), v(1), v(2));
+	      Transformation<3> multi(Vec<3>(0,0,0));
+	      Transformation<3> ht;
+
+	      Solid * hsol = sol1;
+	      for (int i = 1; i <= n; i++)
+		{
+		  Solid * nsol = sol1 -> Copy(*geom);
+
+		  nsol -> Transform (multi);
+		  hsol = new Solid (Solid::UNION, hsol, nsol); 
+
+		  ht=multi;
+		  multi.Combine (trans, ht);
+		}
+	      return hsol;
+	    }
+
+
 	  default:
 	    {
 	      scan.Error (string ("unknown primary ") + scan.GetStringValue());
@@ -826,6 +718,7 @@ namespace netgen
   }
 
 
+
   Solid * ParseTerm (CSGScanner & scan)
   {
     Solid * sol = ParsePrimary(scan);
@@ -838,6 +731,7 @@ namespace netgen
     return sol;
   }
 
+
   Solid * ParseSolid (CSGScanner & scan)
   {
     Solid * sol = ParseTerm(scan);
@@ -896,7 +790,7 @@ namespace netgen
 
   /*
     Main parsing function for CSG geometry
-   */
+  */
   CSGeometry * ParseCSG (istream & istr)
   {
     CSGScanner scan(istr);
@@ -905,9 +799,8 @@ namespace netgen
 
     scan.ReadNext();
     if (scan.GetToken() != TOK_RECO)  // keyword 'algebraic3d'
-      {
-	return 0;
-      }
+      return 0;
+
     scan.ReadNext();
 
     try
@@ -956,10 +849,13 @@ namespace netgen
 		    ParseFlags (scan, flags);
 		    
 		    ParseChar (scan, ';');
+		    if (!geom->GetSolid (name))
+		      scan.Error ("Top-Level-Object "+name+" not defined");
 
 		    int tlonr = 
 		      geom->SetTopLevelObject ((Solid*)geom->GetSolid(name));
 		    TopLevelObject * tlo = geom->GetTopLevelObject (tlonr);
+
 		    if (flags.NumListFlagDefined ("col"))
 		      {
 			const ARRAY<double> & col =
@@ -1025,7 +921,7 @@ namespace netgen
 		      
 		      string name2 = scan.GetStringValue();
 		      scan.ReadNext();
-		      
+
 		      Flags flags;
 		      ParseFlags (scan, flags);
 		      
@@ -1035,12 +931,17 @@ namespace netgen
 		      ARRAY<int> si1, si2;
 		      geom->GetSolid(name1)->GetSurfaceIndices(si1);
 		      geom->GetSolid(name2)->GetSurfaceIndices(si2);
-		      
+
+		      const TopLevelObject * domain = 
+			geom->GetTopLevelObject (geom->GetSolid(flags.GetStringFlag ("tlo","")));
+
 		      geom->AddIdentification 
 			(new CloseSurfaceIdentification 
 			 (geom->GetNIdentifications()+1, *geom, 
 			  geom->GetSurface (si1[0]), geom->GetSurface (si2[0]),
+			  domain,
 			  flags));
+
 		      break;
 		    }
 		    
@@ -1075,6 +976,86 @@ namespace netgen
 		  }
 		
 	      }
+
+	    else if (scan.GetToken() == TOK_SINGULAR)
+
+	      {
+		
+		scan.ReadNext();
+		switch (scan.GetToken())
+		  {
+		  case TOK_FACE:
+		    {
+		      scan.ReadNext();
+		      
+		      string name1 = scan.GetStringValue();  // tlo
+		      scan.ReadNext();
+		      
+		      string name2 = scan.GetStringValue();
+		      scan.ReadNext();
+		      
+		      Flags flags;
+		      ParseFlags (scan, flags);
+		      
+		      ParseChar (scan, ';');
+		      
+		      const Solid * sol = geom->GetSolid(name2);
+
+		      for (int i = 0; i < geom->GetNTopLevelObjects(); i++)
+			if (name1 == geom->GetTopLevelObject (i)->GetSolid()->Name())
+			  geom->singfaces.Append (new SingularFace (i+1, sol));
+
+		      break;
+		    }
+
+		  case TOK_EDGE:
+		    {
+		      scan.ReadNext();
+		      
+		      string name1 = scan.GetStringValue();
+		      scan.ReadNext();
+		      
+		      string name2 = scan.GetStringValue();
+		      scan.ReadNext();
+		      
+		      Flags flags;
+		      ParseFlags (scan, flags);
+		      
+		      ParseChar (scan, ';');
+		      
+		      const Solid * s1 = geom->GetSolid(name1);
+		      const Solid * s2 = geom->GetSolid(name2);
+		      geom->singedges.Append (new SingularEdge (1, s1, s2));
+		      break;
+		    }
+
+		  case TOK_POINT:
+		    {
+		      scan.ReadNext();
+		      
+		      string name1 = scan.GetStringValue();
+		      scan.ReadNext();
+		      string name2 = scan.GetStringValue();
+		      scan.ReadNext();
+		      string name3 = scan.GetStringValue();
+		      scan.ReadNext();
+		      
+		      Flags flags;
+		      ParseFlags (scan, flags);
+		      
+		      ParseChar (scan, ';');
+		      
+		      const Solid * s1 = geom->GetSolid(name1);
+		      const Solid * s2 = geom->GetSolid(name2);
+		      const Solid * s3 = geom->GetSolid(name3);
+		      geom->singpoints.Append (new SingularPoint (1, s1, s2, s3));
+		      break;
+		    }
+		  default:
+		    scan.Error ("keyword 'face' or 'edge' or 'point' expected");
+		  }
+	      }
+
 	    
 	    else if (scan.GetToken() == TOK_POINT)
 	      {
@@ -1082,11 +1063,7 @@ namespace netgen
 
 		scan.ReadNext();
 		ParseChar (scan, '(');
-		p(0) = ParseNumber (scan);
-		ParseChar (scan, ',');
-		p(1) = ParseNumber (scan);
-		ParseChar (scan, ',');
-		p(2) = ParseNumber (scan);
+		p = Point<3> (ParseVector (scan));
 		ParseChar (scan, ')');
 		ParseChar (scan, ';');
 
@@ -1099,17 +1076,9 @@ namespace netgen
 		
 		scan.ReadNext();
 		ParseChar (scan, '(');
-		p1(0) = ParseNumber (scan);
-		ParseChar (scan, ',');
-		p1(1) = ParseNumber (scan);
-		ParseChar (scan, ',');
-		p1(2) = ParseNumber (scan);
+		p1 = Point<3> (ParseVector (scan));
 		ParseChar (scan, ';');
-		p2(0) = ParseNumber (scan);
-		ParseChar (scan, ',');
-		p2(1) = ParseNumber (scan);
-		ParseChar (scan, ',');
-		p2(2) = ParseNumber (scan);
+		p2 = Point<3> (ParseVector (scan));
 		ParseChar (scan, ')');
 		ParseChar (scan, ';');
 
@@ -1164,20 +1133,21 @@ namespace netgen
     catch (string errstr)
       {
 	cout << "caught error " << errstr << endl;
+	throw NgException (errstr);
       }
 
 
     return geom;
     /*
-    do
+      do
       {
-	scan.ReadNext();
-	if (scan.GetToken() == TOK_STRING)
-	  cout << "found string " << scan.GetStringValue() << endl;
-	else
-	  cout << "token = " << int(scan.GetToken()) << endl;
+      scan.ReadNext();
+      if (scan.GetToken() == TOK_STRING)
+      cout << "found string " << scan.GetStringValue() << endl;
+      else
+      cout << "token = " << int(scan.GetToken()) << endl;
       }
-    while (scan.GetToken() != TOK_END);
+      while (scan.GetToken() != TOK_END);
     */
   }
 
diff --git a/Netgen/libsrc/csg/edgeflw.cpp b/Netgen/libsrc/csg/edgeflw.cpp
index 0e5418ad69..50b425f84f 100644
--- a/Netgen/libsrc/csg/edgeflw.cpp
+++ b/Netgen/libsrc/csg/edgeflw.cpp
@@ -9,17 +9,33 @@ namespace netgen
 
   EdgeCalculation :: 
   EdgeCalculation (const CSGeometry & ageometry,
-		   const ARRAY<SpecialPoint> & aspecpoints)
+		   ARRAY<SpecialPoint> & aspecpoints)
     : geometry(ageometry), specpoints(aspecpoints)
   {
-    ;
+    Box<3> bbox;
+    if (specpoints.Size() >= 1)
+      bbox.Set (specpoints[0].p);
+    else
+      { bbox.Set (Point<3> (0,0,0)); bbox.Add (Point<3> (1,1,1)); }
+    for (int i = 1; i < specpoints.Size(); i++)
+      bbox.Add (specpoints[i].p);
+
+    searchtree = new Point3dTree (bbox.PMin(), bbox.PMax());
+    meshpoint_tree = new Point3dTree (bbox.PMin(), bbox.PMax());
+
+    for (int i = 0; i < specpoints.Size(); i++)
+      searchtree->Insert (specpoints[i].p, i);
   }
 
+  EdgeCalculation :: ~EdgeCalculation()
+  {
+    delete searchtree;
+    delete meshpoint_tree;
+  }
 
 
   void EdgeCalculation :: Calc(double h, Mesh & mesh)
   {
-    (*testout) << "Find edges" << endl;
     PrintMessage (1, "Find edges");
     PushStatus ("Find edges");
 
@@ -27,6 +43,7 @@ namespace netgen
     SplitEqualOneSegEdges (mesh);
     FindClosedSurfaces (h, mesh);
     PrintMessage (3, cntedge, " edges found");
+
     PopStatus ();
   }
 
@@ -36,44 +53,43 @@ namespace netgen
 
   void EdgeCalculation :: CalcEdges1 (double h, Mesh & mesh)
   {
-    ARRAY<SpecialPoint> hsp(specpoints.Size());
-    ARRAY<SpecialPoint> startpoints, endpoints;
+    ARRAY<int> hsp(specpoints.Size());
+    ARRAY<int> glob2hsp(specpoints.Size());
+    ARRAY<int> startpoints, endpoints;
 
-    int i, j, k, l, hi, pos, ep, ne;
+    int pos, ep;
     int layer;
 
-    Vec<3> a1, a2, t, n, m;
-    Point<3> p, np, pnp, hp;
-
-    Segment seg;
+    Point<3> p, np; 
     int pi1, s1, s2;
-    int lastpi, thispi;
 
     ARRAY<Point<3> > edgepoints;
     ARRAY<double> curvelength;
-    int copyedge, copyfromedge, copyedgeidentification;
+    int copyedge, copyfromedge = -1, copyedgeidentification = -1;
 
-    ARRAY<int> locsurfind;
+    ARRAY<int> locsurfind, locind;
 
-    double len, corr, lam;
-    double steplen, cursteplen, loch, hd;
+    // double len, corr, lam;
+    // double steplen, cursteplen, loch, 
+    double hd;
 
     int checkedcopy = 0;
 
-    double size = geometry.MaxSize(); // globflags.GetNumFlag ("maxsize", 500);
-    double epspointdist2 = size * 1e-6; // globflags.GetNumFlag ("epspointdist", size * 1e-6);
-    epspointdist2 = sqr (epspointdist2);
-
-
-    Solid * locsol;
-
+    // double size = geometry.MaxSize(); 
+    // double epspointdist2 = sqr (size) * 1e-12;
+    
 
     // copy special points to work with
-    for (i = 0; i < specpoints.Size(); i++)
-      hsp[i] = specpoints[i];
+    for (int i = 0; i < specpoints.Size(); i++)
+      {
+	hsp[i] = i;
+	glob2hsp[i] = i;
+      }
 
 
     cntedge = 0;
+    INDEX_2_HASHTABLE<int> identification_used(100);  // identification i already used for startpoint j
+
 
     while (hsp.Size())
       {
@@ -89,73 +105,46 @@ namespace netgen
 
 	//      (*testout) << endl;
 
-	for (i = 1; i <= geometry.identifications.Size() && !pi1; i++)
-	  {
-	    for (j = checkedcopy+1; j <= startpoints.Size() && !pi1; j++)
-	      {
+	for (int i = 0; i < geometry.identifications.Size() && !pi1; i++)
+	  for (int j = checkedcopy; j < startpoints.Size() && !pi1; j++)
 
-		if (geometry.identifications.Get(i)->IdentifyableCandidate (startpoints.Get(j)))
+	    if (geometry.identifications[i]->IdentifyableCandidate (specpoints[startpoints[j]]))
+	      
+	      {
+		int pi1cand = 0;
+		double mindist = 1e10;
 		
+		for (int k = 0; k < hsp.Size() && !pi1; k++)
 		  {
-		    int pi1cand = 0;
-		    double mindist = 1e10;
-		  
-		    for (k = 1; k <= hsp.Size() && !pi1; k++)
+		    if (identification_used.Used (INDEX_2(i, startpoints[j])) ||
+			identification_used.Used (INDEX_2(i, hsp[k]))) continue;
+		    
+		    if (geometry.identifications[i]
+			->Identifyable(specpoints[startpoints[j]], specpoints[hsp[k]]) ||
+			geometry.identifications[i]
+			->Identifyable(specpoints[hsp[k]], specpoints[startpoints[j]]))
 		      {
-#ifdef DEVELOP
-			(*testout) << "check kand = " << hsp.Get(k).p 
-				   << ", v = " << hsp.Get(k).v 
-				   << endl;		      
-#endif
-			if (geometry.identifications.Get(i)
-			    ->Identifyable(startpoints.Get(j), hsp.Get(k)) ||
-			    geometry.identifications.Get(i)
-			    ->Identifyable(hsp.Get(k), startpoints.Get(j)))
+			if (Dist (specpoints[startpoints[j]].p, specpoints[hsp[k]].p) < mindist)
 			  {
-
-#ifdef DEVELOP
-			    (*testout) << "identifiable, dist = "
-				       << Dist (startpoints.Get(j).p, hsp.Get(k).p) << endl;
-#endif
-
-			    if (Dist (startpoints.Get(j).p, hsp.Get(k).p) < mindist)
-			      {
-				mindist = Dist (startpoints.Get(j).p, hsp.Get(k).p);
-				pi1cand = k;
-			      }
-			    /*
-			      pi1 = k;
-			      copyedge = 1;
-			      copyfromedge = j;
-			      copyedgeidentification = i;
-			  
-			      (*testout) << "copy edge startpoint from "
-			      << startpoints.Get(j).p << " - " 
-			      << startpoints.Get(j).v 
-			      << " to " 
-			      << hsp.Get(k).p << " - " << hsp.Get(k).v << endl;
-			    */
+			    mindist = Dist (specpoints[startpoints[j]].p, specpoints[hsp[k]].p);
+			    pi1cand = k+1;
 			  }
 		      }
-
-		    if (pi1cand)
-		      {
-			pi1 = pi1cand;
-			copyedge = 1;
-			copyfromedge = j;
-			copyedgeidentification = i;
-#ifdef DEVELOP
-			(*testout) << "copy edge startpoint from "
-				   << startpoints.Get(j).p << " - " 
-				   << startpoints.Get(j).v 
-				   << " to " 
-				   << hsp.Get(pi1).p << " - " << hsp.Get(pi1).v << endl;
-#endif
-		      }
+		  }
+	
+	
+		if (pi1cand)
+		  {
+		    pi1 = pi1cand;
+		    copyedge = 1;
+		    copyfromedge = j+1;
+		    copyedgeidentification = i+1;
+		    
+		    identification_used.Set (INDEX_2(i, startpoints[j]), 1);
+		    identification_used.Set (INDEX_2(i, hsp.Get(pi1)), 1);
 		  }
 	      }
-	  }
-      
+	
       
 	// cannot copy from other ege ?
 	if (!pi1)
@@ -163,9 +152,12 @@ namespace netgen
       
 	// unconditional special point available ?
 	if (!pi1)
-	  for (i = 1; i <= hsp.Size() && pi1 == 0; i++)
-	    if (hsp.Get(i).unconditional == 1)
-	      pi1 = i;
+	  for (int i = 1; i <= hsp.Size(); i++)
+	    if (specpoints[hsp.Get(i)].unconditional == 1)
+	      {
+		pi1 = i;
+		break;
+	      }
  
      
 	if (!pi1)
@@ -174,18 +166,19 @@ namespace netgen
 	    pi1 = 1;
 	  }
 
-	layer = hsp.Get(pi1).GetLayer();
+	layer = specpoints[hsp.Get(pi1)].GetLayer();
       
 
-	if (!hsp.Get(pi1).unconditional)
+	if (!specpoints[hsp.Get(pi1)].unconditional)
 	  {
-	    hsp.Elem(pi1).unconditional = 1;
-	    for (i = 1; i <= hsp.Size(); i++)
-	      if (i != pi1 && Dist (hsp.Get(pi1).p, hsp.Get(i).p) < 1e-8 &&
-		  (hsp.Get(pi1).v + hsp.Get(i).v).Length() < 1e-4)
+	    specpoints[hsp.Elem(pi1)].unconditional = 1;
+	    for (int i = 1; i <= hsp.Size(); i++)
+	      if (i != pi1 && 
+		  Dist (specpoints[hsp.Get(pi1)].p, specpoints[hsp.Get(i)].p) < 1e-8 &&
+		  (specpoints[hsp.Get(pi1)].v + specpoints[hsp.Get(i)].v).Length() < 1e-4)
 		{
 		  // opposite direction
-		  hsp.Elem(i).unconditional = 1;
+		  specpoints[hsp.Elem(i)].unconditional = 1;
 		}
 	  }
 
@@ -194,7 +187,8 @@ namespace netgen
 
 #ifdef DEVELOP
 	(*testout) << "edge nr " << cntedge << endl;
-	(*testout) << "start followedge: p1 = " << hsp.Get(pi1).p << ", v = " << hsp.Get(pi1).v << endl;
+	(*testout) << "start followedge: p1 = " << specpoints[hsp.Get(pi1)].p 
+		   << ", v = " << specpoints[hsp.Get(pi1)].v << endl;
 #endif
 
 	FollowEdge (pi1, ep, pos, hsp, h, mesh,
@@ -208,6 +202,7 @@ namespace netgen
 	  {
 	    // ignore starting point
 	    hsp.DeleteElement (pi1);
+	    cout << "yes, this happens" << endl;
 	    continue;
 	  }
 
@@ -217,81 +212,121 @@ namespace netgen
 
 
 	double elen = 0;
-	for (i = 1; i <= edgepoints.Size()-1; i++)
+	for (int i = 1; i <= edgepoints.Size()-1; i++)
 	  elen += Dist (edgepoints.Get(i), edgepoints.Get(i+1));
 
 
 	int shortedge = 0;
-	for (i = 1; i <= geometry.identifications.Size(); i++)
-	  if (geometry.identifications.Get(i)->ShortEdge(hsp.Get(pi1), hsp.Get(ep)))
+	for (int i = 1; i <= geometry.identifications.Size(); i++)
+	  if (geometry.identifications.Get(i)->ShortEdge(specpoints[hsp.Get(pi1)], specpoints[hsp.Get(ep)]))
 	    shortedge = 1;
-	(*testout) << "shortedge = " << shortedge << endl;
+	// (*testout) << "shortedge = " << shortedge << endl;
 
 
 	if (!shortedge)
 	  {
-	    mesh.RestrictLocalHLine (Point3d (hsp.Get(pi1).p), 
-				     Point3d (hsp.Get(ep).p), 
+	    mesh.RestrictLocalHLine (Point3d (specpoints[hsp.Get(pi1)].p), 
+				     Point3d (specpoints[hsp.Get(ep)].p), 
 				     elen / mparam.segmentsperedge);
 	  }
       
-	s1 = hsp.Get(pi1).s1;
-	s2 = hsp.Get(pi1).s2;
+	s1 = specpoints[hsp.Get(pi1)].s1;
+	s2 = specpoints[hsp.Get(pi1)].s2;
 
 
 	// delete initial, terminal and conditional points
 
 #ifdef DEVELOP
-	(*testout) << "terminal point: p = " << hsp.Get(ep).p << ", v = " << hsp.Get(ep).v << endl;      
+	(*testout) << "terminal point: p = " << specpoints[hsp.Get(ep)].p 
+		   << ", v = " << specpoints[hsp.Get(ep)].v << endl;      
 #endif
+
+	searchtree -> DeleteElement (hsp.Get(ep));
+	searchtree -> DeleteElement (hsp.Get(pi1));
+
 	if (ep > pi1)
 	  {
+	    glob2hsp[hsp[ep-1]] = -1;
+	    glob2hsp[hsp.Last()] = ep-1;
 	    hsp.DeleteElement (ep);
+
+	    glob2hsp[hsp[pi1-1]] = -1;
+	    glob2hsp[hsp.Last()] = pi1-1;
 	    hsp.DeleteElement (pi1);
 	  }
 	else
 	  {
+	    glob2hsp[hsp[pi1-1]] = -1;
+	    glob2hsp[hsp.Last()] = pi1-1;
 	    hsp.DeleteElement (pi1);
+
+	    glob2hsp[hsp[ep-1]] = -1;
+	    glob2hsp[hsp.Last()] = ep-1;
 	    hsp.DeleteElement (ep);
 	  }
 
 
-	for (j = 1; j <= edgepoints.Size()-1; j++)
+	for (int j = 1; j <= edgepoints.Size()-1; j++)
 	  {
 	    p = edgepoints.Get(j);
 	    np = Center (p, edgepoints.Get(j+1));
-	    hd = Dist2 (p, np);
+	    hd = Dist (p, np);
  
-	    for (i = 1; i <= hsp.Size(); i++)
-	      if ( hsp.Get(i).HasSurfaces (s1, s2) &&
-		   hsp.Get(i).unconditional == 0 &&
-		   Dist2 (np, hsp.Get(i).p) < 1.2 * hd)
+
+	    Box<3> boxp (np - (1.2 * hd) * Vec<3> (1, 1, 1),
+			 np + (1.2 * hd) * Vec<3> (1, 1, 1));
+	    searchtree -> GetIntersecting (boxp.PMin(), boxp.PMax(), locind);	    
+
+	    for (int i = 0; i < locind.Size(); i++)
+	      {
+		if ( specpoints[locind[i]].HasSurfaces (s1, s2) &&
+		     specpoints[locind[i]].unconditional == 0)
+		  {
+		    searchtree -> DeleteElement (locind[i]);
+
+		    int li = glob2hsp[locind[i]];
+		    glob2hsp[locind[i]] = -1;
+		    glob2hsp[hsp.Last()] = li;
+		    hsp.Delete (li);
+		  }
+	      }
+
+
+	    /*
+	    for (int i = 1; i <= hsp.Size(); i++)
+	      if ( specpoints[hsp.Get(i)].HasSurfaces (s1, s2) &&
+		   specpoints[hsp.Get(i)].unconditional == 0 &&
+		   Dist2 (np, specpoints[hsp.Get(i)].p) < 1.2 * hd)
 		{
+		  searchtree -> DeleteElement (hsp.Get(i)+1);
 		  hsp.DeleteElement (i);
 		  i--;
 		}
+	    */
 	  }
 
       
 	ARRAY<Segment> refedges;
-	ARRAY<int> refedgesinv;
+	ARRAY<bool> refedgesinv;
       
 
 	AnalyzeEdge (s1, s2, pos, layer,
 		     edgepoints,
 		     refedges, refedgesinv);
 
-	for (i = 1; i <= refedges.Size(); i++)
-	  refedges.Elem(i).edgenr = cntedge;
+	for (int i = 0; i < refedges.Size(); i++)
+	  refedges[i].edgenr = cntedge;
+
 
+	
 
 #ifdef DEVELOP
 	(*testout) << "edge " << cntedge << endl
-		   << "startp: " << startpoints.Last().p 
-		   << ", v = " << startpoints.Last().v << endl
-		   << "copy = " << copyedge << endl
+		   << "startp: " << specpoints[startpoints.Last()].p 
+		   << ", v = " << specpoints[startpoints.Last()].v << endl
+	  // << "copy = " << copyedge << endl
 		   << refedges.Size() << " refedges: ";
-	for (i = 1; i <= refedges.Size(); i++)
+	for (int i = 1; i <= refedges.Size(); i++)
 	  (*testout) << " " << refedges.Get(i).si;
 	(*testout) << endl;
 	(*testout) << "inv[1] = " << refedgesinv.Get(1) << endl;
@@ -299,7 +334,7 @@ namespace netgen
       
 	if (!copyedge)
 	  {
-	    int oldnseg = mesh.GetNSeg();
+	    // int oldnseg = mesh.GetNSeg();
 
 	    if (!shortedge)
 	      StoreEdge (refedges, refedgesinv, 
@@ -310,8 +345,8 @@ namespace netgen
 
 
 	    /*
-	      for (i = oldnseg+1; i <= mesh.GetNSeg(); i++)
-	      for (j = 1; j <= oldnseg; j++)
+	      for (int i = oldnseg+1; i <= mesh.GetNSeg(); i++)
+	      for (int j = 1; j <= oldnseg; j++)
 	      {
 	      const Point<3> & l1p1 = mesh.Point (mesh.LineSegment(i).p1);
 	      const Point<3> & l1p2 = mesh.Point (mesh.LineSegment(i).p2);
@@ -332,8 +367,8 @@ namespace netgen
 	  {
 	    CopyEdge (refedges, refedgesinv,
 		      copyfromedge, 
-		      startpoints.Get(copyfromedge).p,
-		      endpoints.Get(copyfromedge).p,
+		      specpoints[startpoints.Get(copyfromedge)].p,
+		      specpoints[endpoints.Get(copyfromedge)].p,
 		      edgepoints.Get(1), edgepoints.Last(),
 		      copyedgeidentification, 
 		      layer,
@@ -345,6 +380,8 @@ namespace netgen
 
 
 
+
+
   /*
     If two or more edges share the same initial and end-points,
     then they need at least two segments 
@@ -352,7 +389,7 @@ namespace netgen
   void EdgeCalculation ::
   SplitEqualOneSegEdges (Mesh & mesh)
   {
-    int i, j, k;
+    //    int i, j;
     SegmentIndex si;
     PointIndex pi;
 
@@ -369,10 +406,13 @@ namespace netgen
 	  osedges.Elem(seg.edgenr)--;
       }
 
+    //    (*testout) << "osedges  = " << osedges << endl;
+
     // flag one segment edges
-    for (i = 0; i < cntedge; i++)
+    for (int i = 0; i < cntedge; i++)
       osedges[i] = (osedges[i] > 0) ? 1 : 0;
 
+    //    (*testout) << "osedges, now  = " << osedges << endl;
 
     for (si = 0; si < mesh.GetNSeg(); si++)
       {
@@ -394,9 +434,9 @@ namespace netgen
 
     // one edge 1 segment, other 2 segments 
     // yes, it happens !
-  
-    for (i = 1; i <= osedgesht.GetNBags(); i++)
-      for (j = 1; j <= osedgesht.GetBagSize(i); j++)
+    point_on_edge_problem = 0;
+    for (int i = 1; i <= osedgesht.GetNBags(); i++)
+      for (int j = 1; j <= osedgesht.GetBagSize(i); j++)
 	{
 	  INDEX_2 i2; 
 	  int val;
@@ -420,9 +460,10 @@ namespace netgen
 		    Point<3> hp = p1 + lam * v;
 		    if (Dist (p, hp) < 1e-4 * vlen)
 		      {
-			PrintSysError ("Point on edge !!!");
-			cout << "seg: " << i2 << ", p = " << k << endl;
+			PrintWarning ("Point on edge !!!");
+			cout << "seg: " << i2 << ", p = " << pi << endl;
 			osedgesht.Set (i2, 2);		      
+			point_on_edge_problem = 1;
 		      }
 		  }
 	      }
@@ -453,12 +494,13 @@ namespace netgen
 
 		osedges.Elem(seg.edgenr) = 
 		  mesh.AddPoint (newp, mesh[PointIndex(seg.p1)].GetLayer());
+		meshpoint_tree -> Insert (newp, osedges.Elem(seg.edgenr));
 	      }
 	  }
       }
 
 
-    for (i = 1; i <= nseg; i++)
+    for (int i = 1; i <= nseg; i++)
       {
 	Segment & seg = mesh.LineSegment (i);
 	if (seg.edgenr >= 1 && seg.edgenr <= cntedge)
@@ -479,35 +521,35 @@ namespace netgen
 
   void EdgeCalculation :: 
   FollowEdge (int pi1, int & ep, int & pos,
-	      const ARRAY<SpecialPoint> & hsp,
+	      const ARRAY<int> & hsp,
 	      double h, const Mesh & mesh,
 	      ARRAY<Point<3> > & edgepoints,
 	      ARRAY<double> & curvelength)
   {
-    int i, j, s1, s2;
+    int s1, s2;
     double len, steplen, cursteplen, loch;
     Point<3> p, np, pnp;
     Vec<3> a1, a2, t;
 
+    ARRAY<int> locind;
 
-    double size = geometry.MaxSize();  // globflags.GetNumFlag ("maxsize", 500);
-    double epspointdist2 = size * 1e-6; // globflags.GetNumFlag ("epspointdist", size * 1e-6);
+    double size = geometry.MaxSize();  
+    double epspointdist2 = size * 1e-6;
     epspointdist2 = sqr (epspointdist2);
     int uselocalh = mparam.uselocalh;
 
 
-
-    s1 = hsp.Get(pi1).s1;
-    s2 = hsp.Get(pi1).s2;
+    s1 = specpoints[hsp.Get(pi1)].s1;
+    s2 = specpoints[hsp.Get(pi1)].s2;
   
-    p = hsp.Get(pi1).p;
+    p = specpoints[hsp.Get(pi1)].p;
     geometry.GetSurface(s1) -> CalcGradient (p, a1);
     geometry.GetSurface(s2) -> CalcGradient (p, a2);
 
     t = Cross (a1, a2);
     t.Normalize();
 
-    pos = (hsp.Get(pi1).v * t) > 0;
+    pos = (specpoints[hsp.Get(pi1)].v * t) > 0;
     if (!pos) t *= -1;
 
   
@@ -534,7 +576,7 @@ namespace netgen
 	if (multithread.terminate)
 	  return;
       
-	if (fabs (p(0)) + fabs (p(1)) + fabs (p(2)) > 10000)
+	if (fabs (p(0)) + fabs (p(1)) + fabs (p(2)) > 100000)
 	  {
 	    ep = 0;
 	    PrintWarning ("Give up line");
@@ -559,15 +601,6 @@ namespace netgen
       
  
 	np = pnp;
-      
-#ifdef MYGRAPH
-	if (silentflag <= 2)
-	  {
-	    MyLine3D (p, np, rot);
-	    MyDraw ();
-	  }
-#endif      
-
 	ep = 0;
       
 	double hvtmin = 1.5 * cursteplen;
@@ -575,19 +608,68 @@ namespace netgen
 	Box<3> boxp (p - (2 * cursteplen) * Vec<3> (1, 1, 1),
 		     p + (2 * cursteplen) * Vec<3> (1, 1, 1));
 
-	for (i = 1; i <= hsp.Size(); i++)
-	  //        if ( i != pi1 && hsp[i].HasSurfaces (s1, s2) )
+	searchtree -> GetIntersecting (boxp.PMin(), boxp.PMax(), locind);
+	
+	for (int i = 0; i < locind.Size(); i++)
 	  {
-	    if (!boxp.IsIn (hsp.Get(i).p))
+	    Vec<3> hv = specpoints[locind[i]].p - p;
+	    if (hv.Length2() > 9 * cursteplen * cursteplen)
 	      continue;
+
+	    double hvt = hv * t;
+	    hv -= hvt * t;
 	  
-	    Vec<3> hv = hsp.Get(i).p - p;
+	    if (hv.Length() < 0.2 * cursteplen &&
+		hvt > 0 && 
+		//		  hvt < 1.5 * cursteplen &&
+		hvt < hvtmin && 
+		specpoints[locind[i]].unconditional == 1 &&
+		(specpoints[locind[i]].v + t).Length() < 0.4  ) 
+	      {
+		Point<3> hep = specpoints[locind[i]].p;
+		ProjectToEdge (geometry.GetSurface(s1), 
+			       geometry.GetSurface(s2), hep);            
+	      
+	      
+		if (Dist2 (hep, specpoints[locind[i]].p) < epspointdist2 )
+		  {
+		    geometry.GetSurface(s1) -> CalcGradient (hep, a1);
+		    geometry.GetSurface(s2) -> CalcGradient (hep, a2);
+		    Vec<3> ept = Cross (a1, a2);
+		    ept /= ept.Length();
+		    if (!pos) ept *= -1;
+		  
+		    if ( (specpoints[locind[i]].v + ept).Length() < 1e-4 )
+		      {
+			np = specpoints[locind[i]].p;
+
+			for (int jj = 0; jj < hsp.Size(); jj++)
+			  if (hsp[jj] == locind[i])
+			    ep = jj+1;
+
+			if (!ep) 
+			  cerr << "endpoint not found" << endl;
+			  //			ep = i;
+			hvtmin = hvt;
+			//			  break;
+		      }
+		  }
+	      }
+	  }
+
+
+
+
+	/*
+	for (int i = 1; i <= hsp.Size(); i++)
+	  {
+	    if (!boxp.IsIn (specpoints[hsp.Get(i)].p))
+	      continue;
+	  
+	    Vec<3> hv = specpoints[hsp.Get(i)].p - p;
 	    if (hv.Length2() > 9 * cursteplen * cursteplen)
 	      continue;
-	    /*
-	      if (!hsp.Get(i).HasSurfaces (s1, s2))
-	      continue;                  // test for dalibor-problem
-	    */
+
 	    double hvt = hv * t;
 	    hv -= hvt * t;
 	  
@@ -595,15 +677,15 @@ namespace netgen
 		hvt > 0 && 
 		//		  hvt < 1.5 * cursteplen &&
 		hvt < hvtmin && 
-		hsp.Get(i).unconditional == 1 &&
-		(hsp.Get(i).v + t).Length() < 0.4  ) 
+		specpoints[hsp.Get(i)].unconditional == 1 &&
+		(specpoints[hsp.Get(i)].v + t).Length() < 0.4  ) 
 	      {
-		Point<3> hep = hsp.Get(i).p;
+		Point<3> hep = specpoints[hsp.Get(i)].p;
 		ProjectToEdge (geometry.GetSurface(s1), 
 			       geometry.GetSurface(s2), hep);            
 	      
 	      
-		if (Dist2 (hep, hsp.Get(i).p) < epspointdist2 )
+		if (Dist2 (hep, specpoints[hsp.Get(i)].p) < epspointdist2 )
 		  {
 		    geometry.GetSurface(s1) -> CalcGradient (hep, a1);
 		    geometry.GetSurface(s2) -> CalcGradient (hep, a2);
@@ -611,9 +693,9 @@ namespace netgen
 		    ept /= ept.Length();
 		    if (!pos) ept *= -1;
 		  
-		    if ( (hsp.Get(i).v + ept).Length() < 1e-4 )
+		    if ( (specpoints[hsp.Get(i)].v + ept).Length() < 1e-4 )
 		      {
-			np = hsp.Get(i).p;
+			np = specpoints[hsp.Get(i)].p;
 			ep = i;
 			hvtmin = hvt;
 			//			  break;
@@ -621,6 +703,7 @@ namespace netgen
 		  }
 	      }
 	  }
+	*/
 
 	loch = min2 (geometry.GetSurface(s1) -> LocH (np, 3, 1, h), 
 		     geometry.GetSurface(s2) -> LocH (np, 3, 1, h));
@@ -658,7 +741,7 @@ namespace netgen
   AnalyzeEdge (int s1, int s2, int pos, int layer,
 	       const ARRAY<Point<3> > & edgepoints,
 	       ARRAY<Segment> & refedges,
-	       ARRAY<int> & refedgesinv)
+	       ARRAY<bool> & refedgesinv)
   {
     int i, j, k, l;
     int hi;
@@ -681,21 +764,24 @@ namespace netgen
       (*testout) << "Analyze edge: " << pi1 << " - " << pi2 << ", pts = " << edgepoints.Size() << endl;
       (*testout) << "p1 = " << edgepoints.Get(1) << " pl = " << edgepoints.Last() << endl;
     */
-    int debug = 0;
+    bool debug = 0;
     /*
-      Dist2 (Point<3> (2.69642, 1.1866, 2.03), edgepoints.Get(1)) < 1e-6 ||
-      Dist2 (Point<3> (2.69642, 1.1866, 2.03), edgepoints.Last()) < 1e-6;
+      Dist2 (Point<3> (1.824, -0.104, -0.95), edgepoints.Get(1)) < 1e-6 ||
+      Dist2 (Point<3> (1.824, -0.104, -0.95), edgepoints.Last()) < 1e-6 ||
+      Dist2 (Point<3> (1.72149, -0.26069, -0.95), edgepoints.Get(1)) < 1e-6 ||
+      Dist2 (Point<3> (1.72149, -0.26069, -0.95), edgepoints.Last()) < 1e-6;
     */
 
     if (debug)
       {
-	//      (*testout) << "tubious edge !!!" << endl;
+	(*testout) << "tubious edge !!!" << endl;
+	(*testout) << "edgepoints = " << edgepoints << endl;
 	(*testout) << "s1, s2 = " << s1 << " - " << s2 << endl;
       }
 
     refedges.SetSize(0);
     refedgesinv.SetSize(0);
-    hp = Center (edgepoints.Get(1), edgepoints.Get(2));
+    hp = Center (edgepoints[0], edgepoints[1]);
     ProjectToEdge (geometry.GetSurface(s1), geometry.GetSurface(s2), hp);
 
     geometry.GetSurface(s1) -> CalcGradient (hp, a1);
@@ -704,12 +790,8 @@ namespace netgen
     t.Normalize();
     if (!pos) t *= -1;    
   
-    (*testout) << "t = " << t << endl;
-
     for (i = 0; i < geometry.GetNTopLevelObjects(); i++)
       {
-	(*testout) << "layer = " << layer 
-		   << ", tlo-layer = " << geometry.GetTopLevelObject(i)->GetLayer() << endl;
 	if (geometry.GetTopLevelObject(i)->GetLayer() != layer) 
 	  continue;
       
@@ -717,6 +799,7 @@ namespace netgen
 	const Surface * surf = geometry.GetTopLevelObject(i)->GetSurface();
 
 	sol -> TangentialSolid (hp, locsol);
+
 	if (!locsol) continue;
 
 	BoxSphere<3> boxp (hp, hp);
@@ -730,7 +813,6 @@ namespace netgen
 
 	locsol -> CalcSurfaceInverse ();
       
-
 	if (!surf)
 	  {
 	    locsol -> GetSurfaceIndices (locsurfind);
@@ -762,7 +844,7 @@ namespace netgen
 	for (j = locsurfind.Size()-1; j >= 0; j--)
 	  if (fabs (geometry.GetSurface(locsurfind[j])
 		    ->CalcFunctionValue (hp) ) > 1e-6)
-	    locsurfind.DeleteElement(j+1);
+	    locsurfind.Delete(j);
       
 	if (debug)
 	  (*testout) << locsurfind.Size() << " faces on hp" << endl;
@@ -775,7 +857,7 @@ namespace netgen
 	    Vec<3> rn;
 
 	    // n is outer normal to solid
-	    geometry.GetSurface(lsi) -> GetNormalVector (hp, n);
+	    n = geometry.GetSurface(lsi) -> GetNormalVector (hp);
 	    if (geometry.GetSurface (lsi)->Inverse())
 	      n *= -1;
 	  
@@ -788,13 +870,15 @@ namespace netgen
 	      }
 	  
 	    // rn is normal to class representant
-	    geometry.GetSurface(rlsi) -> GetNormalVector (hp, rn);
-	  
+	    rn = geometry.GetSurface(rlsi) -> GetNormalVector (hp);
+
 	    int sameasref = ((n * rn) > 0);
 	  
 	    m = Cross (t, rn);
 	    m.Normalize();
 	  
+	    if (debug)
+	      (*testout) << "m = " << m << endl;
 
 	    for (k = 1; k <= 2; k ++)
 	      {
@@ -802,8 +886,8 @@ namespace netgen
 	      
 		if (debug)
 		  {
-		    (*testout) << "onface(" << hp << ", " << m << ")= " 
-			       << locsol->OnFace (hp, m);
+		    (*testout) << "onface(" << hp << ", " << m << ")= " << flush;
+		    (*testout) << locsol->OnFace (hp, m) << flush;
 		    (*testout) << " vec2in = "
 			       << locsol -> VectorIn2 (hp, m, n) << " and " 
 			       << locsol -> VectorIn2 (hp, m, -1 * n) << endl;
@@ -813,6 +897,8 @@ namespace netgen
 		if (locsol -> VectorIn2 (hp, m, n) == 0 &&
 		    locsol -> VectorIn2 (hp, m, -1 * n) == 1)
 		  {
+		    if (debug)
+		      (*testout) << "is true" << endl;
 		    hi = 0;
 		    for (l = 1; l <= refedges.Size(); l++)
 		      {
@@ -854,6 +940,11 @@ namespace netgen
 				 << ", refedgenr = " << hi
 				 << endl;
 		  }
+		else
+		  {
+		    if (debug)
+		      (*testout) << "is false" << endl;
+		  }
 		m *= -1;
 	      } 
 	  }
@@ -865,7 +956,7 @@ namespace netgen
 
   void EdgeCalculation :: 
   StoreEdge (const ARRAY<Segment> & refedges,
-	     const ARRAY<int> & refedgesinv,
+	     const ARRAY<bool> & refedgesinv,
 	     const ARRAY<Point<3> > & edgepoints,
 	     const ARRAY<double> & curvelength,
 	     int layer,
@@ -897,6 +988,8 @@ namespace netgen
     // generate initial point
     p = edgepoints.Get(1);
     lastpi = -1;
+
+    /*
     for (pi = PointIndex::BASE; 
 	 pi < mesh.GetNP()+PointIndex::BASE; pi++)
       if (Dist (mesh[pi], p) < 1e-6)
@@ -904,10 +997,20 @@ namespace netgen
 	  lastpi = pi;
 	  break;
 	}
+    */
+    ARRAY<int> locsearch;
+    meshpoint_tree -> GetIntersecting (p-Vec<3> (1e-6, 1e-6, 1e-6),
+				       p+Vec<3> (1e-6, 1e-6, 1e-6), locsearch);
+    if (locsearch.Size())
+      lastpi = locsearch[0];
+				       
 
-    if (lastpi == -1)
-      lastpi = mesh.AddPoint (p, layer);
 
+    if (lastpi == -1)
+      {
+	lastpi = mesh.AddPoint (p, layer);
+	meshpoint_tree -> Insert (p, lastpi); 
+      }
   
     j = 1;
     for (i = 1; i <= ne; i++)
@@ -924,15 +1027,25 @@ namespace netgen
       
 	thispi = -1;
 	if (i == ne)
+	  {
+	    /*
 	  for (pi = PointIndex::BASE; 
 	       pi < mesh.GetNP()+PointIndex::BASE; pi++)
 	    if (Dist(mesh[pi], np) < 1e-6)
 	      thispi = pi;
-      
+	    */
+	    
+	    meshpoint_tree -> GetIntersecting (np-Vec<3> (1e-6, 1e-6, 1e-6),
+					       np+Vec<3> (1e-6, 1e-6, 1e-6), locsearch);
+	    if (locsearch.Size())
+	      thispi = locsearch[0];
+	  }
+
 	if (thispi == -1)
 	  {
 	    ProjectToEdge (surf1, surf2, np);
 	    thispi = mesh.AddPoint (np, layer);
+	    meshpoint_tree -> Insert (np, thispi);
 	  }
 
 	for (k = 1; k <= refedges.Size(); k++)
@@ -1005,7 +1118,7 @@ namespace netgen
 
   void EdgeCalculation :: 
   StoreShortEdge (const ARRAY<Segment> & refedges,
-		  const ARRAY<int> & refedgesinv,
+		  const ARRAY<bool> & refedgesinv,
 		  const ARRAY<Point<3> > & edgepoints,
 		  const ARRAY<double> & curvelength,
 		  int layer,
@@ -1013,9 +1126,8 @@ namespace netgen
   {
   
     // Calculate optimal element-length
-    int i, j, k;
     PointIndex pi;
-    int ne;
+    // int ne;
     Segment seg;
 
     /*
@@ -1048,7 +1160,11 @@ namespace netgen
 	  break;
 	}
 
-    if (pi1 == -1) pi1 = mesh.AddPoint (p, layer);
+    if (pi1 == -1) 
+      {
+	pi1 = mesh.AddPoint (p, layer);
+	meshpoint_tree -> Insert (p, pi1);
+      }
 
     p = edgepoints.Last();
     PointIndex pi2 = -1;
@@ -1060,7 +1176,11 @@ namespace netgen
 	  pi2 = pi;
 	  break;
 	}
-    if (pi2==-1) pi2 = mesh.AddPoint (p, layer);
+    if (pi2==-1) 
+      {
+	pi2 = mesh.AddPoint (p, layer);
+	meshpoint_tree -> Insert (p, pi2);
+      }
 
     /*
   
@@ -1090,7 +1210,7 @@ namespace netgen
     }
     */
   
-    for (k = 1; k <= refedges.Size(); k++)
+    for (int k = 1; k <= refedges.Size(); k++)
       {
 	if (refedgesinv.Get(k))
 	  {
@@ -1125,7 +1245,7 @@ namespace netgen
 
   void EdgeCalculation :: 
   CopyEdge (const ARRAY<Segment> & refedges,
-	    const ARRAY<int> & refedgesinv,
+	    const ARRAY<bool> & refedgesinv,
 	    int copyfromedge, 
 	    const Point<3> & fromstart, const Point<3> & fromend,
 	    const Point<3> & tostart, const Point<3> & toend,
@@ -1133,11 +1253,11 @@ namespace netgen
 	    int layer,
 	    Mesh & mesh)
   {
-    int i, j, k;
+    int k;
     PointIndex pi;
 
     // copy start and end points
-    for (i = 1; i <= 2; i++)
+    for (int i = 1; i <= 2; i++)
       {
 	Point<3> fromp =
 	  (i == 1) ? fromstart : fromend;
@@ -1156,7 +1276,10 @@ namespace netgen
 	  }
 
 	if (topi == -1)
-	  topi = mesh.AddPoint (top, layer);
+	  {
+	    topi = mesh.AddPoint (top, layer);
+	    meshpoint_tree -> Insert (top, topi);
+	  }
 
 	const Identification & csi = 
 	  (*geometry.identifications.Get(copyedgeidentification));
@@ -1180,7 +1303,7 @@ namespace netgen
       }
 
     int oldns = mesh.GetNSeg();
-    for (i = 1; i <= oldns; i++)
+    for (int i = 1; i <= oldns; i++)
       {
 	// real copy, since array might be reallocated !!
 	const Segment oldseg = mesh.LineSegment(i);
@@ -1201,7 +1324,7 @@ namespace netgen
 
 	for (k = 1; k <= refedges.Size(); k++)
 	  {
-	    int inv = refedgesinv.Get(k);
+	    bool inv = refedgesinv.Get(k);
 
 	    // other edge is inverse
 	    if (oldseg.seginfo == 1)
@@ -1305,7 +1428,7 @@ namespace netgen
 	  {
 	    const Surface * s = geometry.GetSurface(i);
 	    p1 = s -> GetSurfacePoint();
-	    s -> GetNormalVector (p1, nv);
+	    nv = s -> GetNormalVector (p1);
 		    
 	    double hloc = 
 	      min2 (s->LocH (p1, 3, 1, h), mesh.GetH(p1));
@@ -1328,6 +1451,8 @@ namespace netgen
 
 	    seg1.surfnr1 = i;
 	    seg2.surfnr1 = i;
+	    seg1.surfnr2 = i;
+	    seg2.surfnr2 = i;
 
 	    for (j = 0; j < nsol; j++)
 	      {
diff --git a/Netgen/libsrc/csg/edgeflw.hpp b/Netgen/libsrc/csg/edgeflw.hpp
index 175d2f164d..beb1f690a9 100644
--- a/Netgen/libsrc/csg/edgeflw.hpp
+++ b/Netgen/libsrc/csg/edgeflw.hpp
@@ -31,21 +31,26 @@ extern void CalcEdges (const CSGeometry & geometry,
 class EdgeCalculation
 {
   const CSGeometry & geometry;
-  const ARRAY<SpecialPoint> & specpoints;
+  ARRAY<SpecialPoint> & specpoints;
+  Point3dTree * searchtree;
+  Point3dTree * meshpoint_tree;
   int cntedge;
 public:
   EdgeCalculation (const CSGeometry & ageometry,
-		   const ARRAY<SpecialPoint> & aspecpoints);
+		   ARRAY<SpecialPoint> & aspecpoints);
+
+  ~EdgeCalculation();
 
   void Calc(double h, Mesh & mesh);
 
 
 private:
   void CalcEdges1 (double h, Mesh & mesh);
-
+  
 
   void FollowEdge (int pi1, int & ep, int & pos,
-		   const ARRAY<SpecialPoint> & hsp,
+		   // const ARRAY<SpecialPoint> & hsp,
+		   const ARRAY<int> & hsp,
 		   double h, const Mesh & mesh,
 		   ARRAY<Point<3> > & edgepoints,
 		   ARRAY<double> & curvelength);
@@ -54,24 +59,24 @@ private:
   void AnalyzeEdge (int s1, int s2, int pos, int layer,
 		    const ARRAY<Point<3> > & edgepoints,
 		    ARRAY<Segment> & refedges,
-		    ARRAY<int> & refedgesinv);
+		    ARRAY<bool> & refedgesinv);
 
   void StoreEdge (const ARRAY<Segment> & refedges,
-		  const ARRAY<int> & refedgesinv,
+		  const ARRAY<bool> & refedgesinv,
 		  const ARRAY<Point<3> > & edgepoints,
 		  const ARRAY<double> & curvelength,
 		  int layer,
 		  Mesh & mesh);
 
   void StoreShortEdge (const ARRAY<Segment> & refedges,
-		       const ARRAY<int> & refedgesinv,
+		       const ARRAY<bool> & refedgesinv,
 		       const ARRAY<Point<3> > & edgepoints,
 		       const ARRAY<double> & curvelength,
 		       int layer,
 		       Mesh & mesh);
 
   void CopyEdge (const ARRAY<Segment> & refedges,
-		 const ARRAY<int> & refedgesinv,
+		 const ARRAY<bool> & refedgesinv,
 		 int copyfromedge, 
 		 const Point<3> & fromstart, const Point<3> & fromend,
 		 const Point<3> & tostart, const Point<3> & toend,
@@ -83,6 +88,10 @@ private:
   void SplitEqualOneSegEdges (Mesh & mesh);
   void FindClosedSurfaces (double h, Mesh & mesh);
 
+
+public:
+  bool point_on_edge_problem;
+
 };
 
 
diff --git a/Netgen/libsrc/csg/edgeflw2.cpp b/Netgen/libsrc/csg/edgeflw2.cpp
new file mode 100644
index 0000000000..87239c05bb
--- /dev/null
+++ b/Netgen/libsrc/csg/edgeflw2.cpp
@@ -0,0 +1,1404 @@
+#include <mystdlib.h>
+#include <meshing.hpp>
+#include <csg.hpp>
+
+#undef DEVELOP
+
+namespace netgen
+{
+
+  EdgeCalculation :: 
+  EdgeCalculation (const CSGeometry & ageometry,
+		   const ARRAY<SpecialPoint> & aspecpoints)
+    : geometry(ageometry), specpoints(aspecpoints)
+  {
+    ;
+  }
+
+
+
+  void EdgeCalculation :: Calc(double h, Mesh & mesh)
+  {
+    PrintMessage (1, "Find edges");
+    PushStatus ("Find edges");
+
+    CalcEdges1 (h, mesh);
+    SplitEqualOneSegEdges (mesh);
+    FindClosedSurfaces (h, mesh);
+    PrintMessage (3, cntedge, " edges found");
+
+    PopStatus ();
+  }
+
+
+
+
+  void EdgeCalculation :: CalcEdges1 (double h, Mesh & mesh)
+  {
+    ARRAY<SpecialPoint> hsp(specpoints.Size());
+    ARRAY<SpecialPoint> startpoints, endpoints;
+
+    int i, j, k, l, hi, pos, ep, ne;
+    int layer;
+
+    Vec<3> a1, a2, t, n, m;
+    Point<3> p, np, pnp, hp;
+
+    Segment seg;
+    int pi1, s1, s2;
+    int lastpi, thispi;
+
+    ARRAY<Point<3> > edgepoints;
+    ARRAY<double> curvelength;
+    int copyedge, copyfromedge, copyedgeidentification;
+
+    ARRAY<int> locsurfind;
+
+    double len, corr, lam;
+    double steplen, cursteplen, loch, hd;
+
+    int checkedcopy = 0;
+
+    double size = geometry.MaxSize(); // globflags.GetNumFlag ("maxsize", 500);
+    double epspointdist2 = size * 1e-6; // globflags.GetNumFlag ("epspointdist", size * 1e-6);
+    epspointdist2 = sqr (epspointdist2);
+
+
+    Solid * locsol;
+
+
+    // copy special points to work with
+    for (i = 0; i < specpoints.Size(); i++)
+      hsp[i] = specpoints[i];
+
+
+    cntedge = 0;
+
+    while (hsp.Size())
+      {
+	SetThreadPercent(100 - 100 * double (hsp.Size()) / specpoints.Size());
+
+	edgepoints.SetSize (0);
+	curvelength.SetSize (0);
+      
+
+	pi1 = 0;
+	copyedge = 0;
+	// identifyable point available ?
+
+	//      (*testout) << endl;
+
+	for (i = 1; i <= geometry.identifications.Size() && !pi1; i++)
+	  {
+	    for (j = checkedcopy+1; j <= startpoints.Size() && !pi1; j++)
+	      {
+
+		if (geometry.identifications.Get(i)->IdentifyableCandidate (startpoints.Get(j)))
+		
+		  {
+		    int pi1cand = 0;
+		    double mindist = 1e10;
+		  
+		    for (k = 1; k <= hsp.Size() && !pi1; k++)
+		      {
+#ifdef DEVELOP
+			(*testout) << "check kand = " << hsp.Get(k).p 
+				   << ", v = " << hsp.Get(k).v 
+				   << endl;		      
+#endif
+			if (geometry.identifications.Get(i)
+			    ->Identifyable(startpoints.Get(j), hsp.Get(k)) ||
+			    geometry.identifications.Get(i)
+			    ->Identifyable(hsp.Get(k), startpoints.Get(j)))
+			  {
+
+#ifdef DEVELOP
+			    (*testout) << "identifiable, dist = "
+				       << Dist (startpoints.Get(j).p, hsp.Get(k).p) << endl;
+#endif
+
+			    if (Dist (startpoints.Get(j).p, hsp.Get(k).p) < mindist)
+			      {
+				mindist = Dist (startpoints.Get(j).p, hsp.Get(k).p);
+				pi1cand = k;
+			      }
+			    /*
+			      pi1 = k;
+			      copyedge = 1;
+			      copyfromedge = j;
+			      copyedgeidentification = i;
+			  
+			      (*testout) << "copy edge startpoint from "
+			      << startpoints.Get(j).p << " - " 
+			      << startpoints.Get(j).v 
+			      << " to " 
+			      << hsp.Get(k).p << " - " << hsp.Get(k).v << endl;
+			    */
+			  }
+		      }
+
+		    if (pi1cand)
+		      {
+			pi1 = pi1cand;
+			copyedge = 1;
+			copyfromedge = j;
+			copyedgeidentification = i;
+#ifdef DEVELOP
+			(*testout) << "copy edge startpoint from "
+				   << startpoints.Get(j).p << " - " 
+				   << startpoints.Get(j).v 
+				   << " to " 
+				   << hsp.Get(pi1).p << " - " << hsp.Get(pi1).v << endl;
+#endif
+		      }
+		  }
+	      }
+	  }
+      
+      
+	// cannot copy from other ege ?
+	if (!pi1)
+	  checkedcopy = startpoints.Size();
+      
+	// unconditional special point available ?
+	if (!pi1)
+	  for (i = 1; i <= hsp.Size() && pi1 == 0; i++)
+	    if (hsp.Get(i).unconditional == 1)
+	      pi1 = i;
+ 
+     
+	if (!pi1)
+	  {
+	    // only unconditional points available, choose first
+	    pi1 = 1;
+	  }
+
+	layer = hsp.Get(pi1).GetLayer();
+      
+
+	if (!hsp.Get(pi1).unconditional)
+	  {
+	    hsp.Elem(pi1).unconditional = 1;
+	    for (i = 1; i <= hsp.Size(); i++)
+	      if (i != pi1 && Dist (hsp.Get(pi1).p, hsp.Get(i).p) < 1e-8 &&
+		  (hsp.Get(pi1).v + hsp.Get(i).v).Length() < 1e-4)
+		{
+		  // opposite direction
+		  hsp.Elem(i).unconditional = 1;
+		}
+	  }
+
+	cntedge++;
+	startpoints.Append (hsp.Get(pi1));
+
+#ifdef DEVELOP
+	(*testout) << "edge nr " << cntedge << endl;
+	(*testout) << "start followedge: p1 = " << hsp.Get(pi1).p << ", v = " << hsp.Get(pi1).v << endl;
+#endif
+
+	FollowEdge (pi1, ep, pos, hsp, h, mesh,
+		    edgepoints, curvelength);
+
+
+	if (multithread.terminate)
+	  return;
+      
+	if (!ep)
+	  {
+	    // ignore starting point
+	    hsp.DeleteElement (pi1);
+	    continue;
+	  }
+
+
+
+	endpoints.Append (hsp.Get(ep));
+
+
+	double elen = 0;
+	for (i = 1; i <= edgepoints.Size()-1; i++)
+	  elen += Dist (edgepoints.Get(i), edgepoints.Get(i+1));
+
+
+	int shortedge = 0;
+	for (i = 1; i <= geometry.identifications.Size(); i++)
+	  if (geometry.identifications.Get(i)->ShortEdge(hsp.Get(pi1), hsp.Get(ep)))
+	    shortedge = 1;
+	(*testout) << "shortedge = " << shortedge << endl;
+
+
+	if (!shortedge)
+	  {
+	    mesh.RestrictLocalHLine (Point3d (hsp.Get(pi1).p), 
+				     Point3d (hsp.Get(ep).p), 
+				     elen / mparam.segmentsperedge);
+	  }
+      
+	s1 = hsp.Get(pi1).s1;
+	s2 = hsp.Get(pi1).s2;
+
+
+	// delete initial, terminal and conditional points
+
+#ifdef DEVELOP
+	(*testout) << "terminal point: p = " << hsp.Get(ep).p << ", v = " << hsp.Get(ep).v << endl;      
+#endif
+	if (ep > pi1)
+	  {
+	    hsp.DeleteElement (ep);
+	    hsp.DeleteElement (pi1);
+	  }
+	else
+	  {
+	    hsp.DeleteElement (pi1);
+	    hsp.DeleteElement (ep);
+	  }
+
+
+	for (j = 1; j <= edgepoints.Size()-1; j++)
+	  {
+	    p = edgepoints.Get(j);
+	    np = Center (p, edgepoints.Get(j+1));
+	    hd = Dist2 (p, np);
+ 
+	    for (i = 1; i <= hsp.Size(); i++)
+	      if ( hsp.Get(i).HasSurfaces (s1, s2) &&
+		   hsp.Get(i).unconditional == 0 &&
+		   Dist2 (np, hsp.Get(i).p) < 1.2 * hd)
+		{
+		  hsp.DeleteElement (i);
+		  i--;
+		}
+	  }
+
+      
+	ARRAY<Segment> refedges;
+	ARRAY<int> refedgesinv;
+      
+
+	AnalyzeEdge (s1, s2, pos, layer,
+		     edgepoints,
+		     refedges, refedgesinv);
+
+	for (i = 1; i <= refedges.Size(); i++)
+	  refedges.Elem(i).edgenr = cntedge;
+
+
+#ifdef DEVELOP
+	(*testout) << "edge " << cntedge << endl
+		   << "startp: " << startpoints.Last().p 
+		   << ", v = " << startpoints.Last().v << endl
+		   << "copy = " << copyedge << endl
+		   << refedges.Size() << " refedges: ";
+	for (i = 1; i <= refedges.Size(); i++)
+	  (*testout) << " " << refedges.Get(i).si;
+	(*testout) << endl;
+	(*testout) << "inv[1] = " << refedgesinv.Get(1) << endl;
+#endif
+      
+	if (!copyedge)
+	  {
+	    int oldnseg = mesh.GetNSeg();
+
+	    if (!shortedge)
+	      StoreEdge (refedges, refedgesinv, 
+			 edgepoints, curvelength, layer, mesh);
+	    else
+	      StoreShortEdge (refedges, refedgesinv, 
+			      edgepoints, curvelength, layer, mesh);
+
+
+	    /*
+	      for (i = oldnseg+1; i <= mesh.GetNSeg(); i++)
+	      for (j = 1; j <= oldnseg; j++)
+	      {
+	      const Point<3> & l1p1 = mesh.Point (mesh.LineSegment(i).p1);
+	      const Point<3> & l1p2 = mesh.Point (mesh.LineSegment(i).p2);
+	      const Point<3> & l2p1 = mesh.Point (mesh.LineSegment(j).p1);
+	      const Point<3> & l2p2 = mesh.Point (mesh.LineSegment(j).p2);
+	      Vec<3> vl1(l1p1, l1p2);
+	      for (double lamk = 0; lamk <= 1; lamk += 0.1)
+	      {
+	      Point<3> l2p = l1p1 + lamk * vl1;
+	      double dist = sqrt (MinDistLP2 (l2p1, l2p2, l2p));
+	      if (dist > 1e-12)
+	      mesh.RestrictLocalH (l2p, 3*dist);
+	      }
+	      }
+	    */
+	  }
+	else
+	  {
+	    CopyEdge (refedges, refedgesinv,
+		      copyfromedge, 
+		      startpoints.Get(copyfromedge).p,
+		      endpoints.Get(copyfromedge).p,
+		      edgepoints.Get(1), edgepoints.Last(),
+		      copyedgeidentification, 
+		      layer,
+		      mesh);
+	  }
+
+      }
+  }
+
+
+
+  /*
+    If two or more edges share the same initial and end-points,
+    then they need at least two segments 
+  */
+  void EdgeCalculation ::
+  SplitEqualOneSegEdges (Mesh & mesh)
+  {
+    int i, j;
+    SegmentIndex si;
+    PointIndex pi;
+
+    ARRAY<int> osedges(cntedge);
+    INDEX_2_HASHTABLE<int> osedgesht (cntedge+1);
+
+    osedges = 2;
+
+    // count segments on edges
+    for (si = 0; si < mesh.GetNSeg(); si++)
+      {
+	const Segment & seg = mesh[si];
+	if (seg.seginfo && seg.edgenr >= 1 && seg.edgenr <= cntedge)
+	  osedges.Elem(seg.edgenr)--;
+      }
+
+    (*testout) << "osedges  = " << osedges << endl;
+
+    // flag one segment edges
+    for (i = 0; i < cntedge; i++)
+      osedges[i] = (osedges[i] > 0) ? 1 : 0;
+
+    (*testout) << "osedges, now  = " << osedges << endl;
+
+    for (si = 0; si < mesh.GetNSeg(); si++)
+      {
+	const Segment & seg = mesh[si];
+	if (seg.seginfo && seg.edgenr >= 1 && seg.edgenr <= cntedge)
+	  {
+	    if (osedges.Get(seg.edgenr))
+	      {
+		INDEX_2 i2(seg.p1, seg.p2);
+		i2.Sort ();
+		if (osedgesht.Used (i2))
+		  osedgesht.Set (i2, 2);
+		else
+		  osedgesht.Set (i2, 1);
+	      }
+	  }
+      }
+
+
+    // one edge 1 segment, other 2 segments 
+    // yes, it happens !
+  
+    for (i = 1; i <= osedgesht.GetNBags(); i++)
+      for (j = 1; j <= osedgesht.GetBagSize(i); j++)
+	{
+	  INDEX_2 i2; 
+	  int val;
+	  osedgesht.GetData (i, j, i2, val);
+
+	  const Point<3> & p1 = mesh[PointIndex(i2.I1())];
+	  const Point<3> & p2 = mesh[PointIndex(i2.I2())];
+	  Vec<3> v = p2 - p1;
+	  double vlen = v.Length();
+	  v /= vlen;
+	  for (pi = PointIndex::BASE; 
+	       pi < mesh.GetNP()+PointIndex::BASE; pi++)
+
+	    if (pi != i2.I1() && pi != i2.I2())
+	      {
+		const Point<3> & p = mesh[pi];
+		Vec<3> v2 = p - p1;
+		double lam = (v2 * v);
+		if (lam > 0 && lam < vlen)
+		  {
+		    Point<3> hp = p1 + lam * v;
+		    if (Dist (p, hp) < 1e-4 * vlen)
+		      {
+			PrintSysError ("Point on edge !!!");
+			cout << "seg: " << i2 << ", p = " << pi << endl;
+			osedgesht.Set (i2, 2);		      
+		      }
+		  }
+	      }
+	}
+
+
+    // insert new points
+    osedges = -1;
+
+    int nseg = mesh.GetNSeg();
+    for (si = 0; si < nseg; si++)
+      {
+	const Segment & seg = mesh[si];
+	if (seg.seginfo && seg.edgenr >= 1 && seg.edgenr <= cntedge)
+	  {
+	    INDEX_2 i2(seg.p1, seg.p2);
+	    i2.Sort ();
+	    if (osedgesht.Used (i2) &&
+		osedgesht.Get (i2) == 2 &&
+		osedges.Elem(seg.edgenr) == -1)
+	      {
+		Point<3> newp = Center (mesh[PointIndex(seg.p1)],
+					mesh[PointIndex(seg.p2)]);
+
+		ProjectToEdge (geometry.GetSurface(seg.surfnr1), 
+			       geometry.GetSurface(seg.surfnr2), 
+			       newp);
+
+		osedges.Elem(seg.edgenr) = 
+		  mesh.AddPoint (newp, mesh[PointIndex(seg.p1)].GetLayer());
+	      }
+	  }
+      }
+
+
+    for (i = 1; i <= nseg; i++)
+      {
+	Segment & seg = mesh.LineSegment (i);
+	if (seg.edgenr >= 1 && seg.edgenr <= cntedge)
+	  {
+	    if (osedges.Get(seg.edgenr) != -1)
+	      {
+		Segment newseg = seg;
+		newseg.p1 = osedges.Get(seg.edgenr);
+		seg.p2 = osedges.Get(seg.edgenr);
+		mesh.AddSegment (newseg);
+	      }
+	  }
+      }
+
+  }
+
+
+
+  void EdgeCalculation :: 
+  FollowEdge (int pi1, int & ep, int & pos,
+	      const ARRAY<SpecialPoint> & hsp,
+	      double h, const Mesh & mesh,
+	      ARRAY<Point<3> > & edgepoints,
+	      ARRAY<double> & curvelength)
+  {
+    int i, j, s1, s2;
+    double len, steplen, cursteplen, loch;
+    Point<3> p, np, pnp;
+    Vec<3> a1, a2, t;
+
+
+    double size = geometry.MaxSize();  
+    double epspointdist2 = size * 1e-6;
+    epspointdist2 = sqr (epspointdist2);
+    int uselocalh = mparam.uselocalh;
+
+
+    s1 = hsp.Get(pi1).s1;
+    s2 = hsp.Get(pi1).s2;
+  
+    p = hsp.Get(pi1).p;
+    geometry.GetSurface(s1) -> CalcGradient (p, a1);
+    geometry.GetSurface(s2) -> CalcGradient (p, a2);
+
+    t = Cross (a1, a2);
+    t.Normalize();
+
+    pos = (hsp.Get(pi1).v * t) > 0;
+    if (!pos) t *= -1;
+
+  
+    edgepoints.Append (p);
+    curvelength.Append (0);
+    len = 0;
+
+    loch = min2 (geometry.GetSurface(s1) -> LocH (p, 3, 1, h), 
+		 geometry.GetSurface(s2) -> LocH (p, 3, 1, h));
+  
+  
+  
+    if (uselocalh)
+      {
+	double lh = mesh.GetH(p);
+	if (lh < loch)
+	  loch = lh;
+      }
+
+    steplen = 0.1 * loch;
+  
+    do
+      {
+	if (multithread.terminate)
+	  return;
+      
+	if (fabs (p(0)) + fabs (p(1)) + fabs (p(2)) > 10000)
+	  {
+	    ep = 0;
+	    PrintWarning ("Give up line");
+	    break;
+	  }
+
+	if (steplen > 0.1 * loch) steplen = 0.1 * loch;
+      
+	steplen *= 2;
+	do
+	  {
+	    steplen *= 0.5;
+	    np = p + steplen * t;
+	    pnp = np;
+	    ProjectToEdge (geometry.GetSurface(s1), 
+			   geometry.GetSurface(s2), pnp);
+	  }
+	while (Dist (np, pnp) > 0.1 * steplen);
+      
+	cursteplen = steplen;
+	if (Dist (np, pnp) < 0.01 * steplen) steplen *= 2;
+      
+ 
+	np = pnp;
+      
+#ifdef MYGRAPH
+	if (silentflag <= 2)
+	  {
+	    MyLine3D (p, np, rot);
+	    MyDraw ();
+	  }
+#endif      
+
+	ep = 0;
+      
+	double hvtmin = 1.5 * cursteplen;
+      
+	Box<3> boxp (p - (2 * cursteplen) * Vec<3> (1, 1, 1),
+		     p + (2 * cursteplen) * Vec<3> (1, 1, 1));
+
+	for (i = 1; i <= hsp.Size(); i++)
+	  // if ( i != pi1 && hsp.Get(i).HasSurfaces (s1, s2) )
+	  {
+	    if (!boxp.IsIn (hsp.Get(i).p))
+	      continue;
+	  
+	    Vec<3> hv = hsp.Get(i).p - p;
+	    if (hv.Length2() > 9 * cursteplen * cursteplen)
+	      continue;
+
+	    /*
+	    if (!hsp.Get(i).HasSurfaces (s1, s2))
+	      continue;                  // test for dalibor-problem
+	    */
+
+	    double hvt = hv * t;
+	    hv -= hvt * t;
+	  
+	    if (hv.Length() < 0.2 * cursteplen &&
+		hvt > 0 && 
+		//		  hvt < 1.5 * cursteplen &&
+		hvt < hvtmin && 
+		hsp.Get(i).unconditional == 1 &&
+		(hsp.Get(i).v + t).Length() < 0.4  ) 
+	      {
+		Point<3> hep = hsp.Get(i).p;
+		ProjectToEdge (geometry.GetSurface(s1), 
+			       geometry.GetSurface(s2), hep);            
+	      
+	      
+		if (Dist2 (hep, hsp.Get(i).p) < epspointdist2 )
+		  {
+		    geometry.GetSurface(s1) -> CalcGradient (hep, a1);
+		    geometry.GetSurface(s2) -> CalcGradient (hep, a2);
+		    Vec<3> ept = Cross (a1, a2);
+		    ept /= ept.Length();
+		    if (!pos) ept *= -1;
+		  
+		    if ( (hsp.Get(i).v + ept).Length() < 1e-4 )
+		      {
+			np = hsp.Get(i).p;
+			ep = i;
+			hvtmin = hvt;
+			//			  break;
+		      }
+		  }
+	      }
+	  }
+
+	loch = min2 (geometry.GetSurface(s1) -> LocH (np, 3, 1, h), 
+		     geometry.GetSurface(s2) -> LocH (np, 3, 1, h));
+
+	if (uselocalh)
+	  {
+	    double lh = mesh.GetH(np);
+	    if (lh < loch)
+	      loch = lh;
+	  }
+      
+      
+	len += Dist (p, np) / loch;
+	edgepoints.Append (np);
+	curvelength.Append (len);
+      
+	p = np;
+      
+	geometry.GetSurface(s1) -> CalcGradient (p, a1);
+	geometry.GetSurface(s2) -> CalcGradient (p, a2);
+	t = Cross (a1, a2);
+	t.Normalize();
+	if (!pos) t *= -1;
+      }
+    while (! ep);
+  }
+
+
+
+
+
+
+
+  void EdgeCalculation :: 
+  AnalyzeEdge (int s1, int s2, int pos, int layer,
+	       const ARRAY<Point<3> > & edgepoints,
+	       ARRAY<Segment> & refedges,
+	       ARRAY<int> & refedgesinv)
+  {
+    int i, j, k, l;
+    int hi;
+    Point<3> hp;
+    Vec<3> t, a1, a2, m, n;
+    Segment seg;
+    Solid * locsol;
+    ARRAY<int> locsurfind;
+
+    /*
+      int pi1 = 0, pi2 = 0;
+      extern Mesh * mesh;
+      for (i = 1; i <= mesh->GetNP(); i++)
+      {
+      if (Dist2 (edgepoints.Get(1), mesh->Point(i)) < 1e-12)
+      pi1 = i;
+      if (Dist2 (edgepoints.Last(), mesh->Point(i)) < 1e-12)
+      pi2 = i;
+      }
+      (*testout) << "Analyze edge: " << pi1 << " - " << pi2 << ", pts = " << edgepoints.Size() << endl;
+      (*testout) << "p1 = " << edgepoints.Get(1) << " pl = " << edgepoints.Last() << endl;
+    */
+    int debug = 0;
+    /*
+      Dist2 (Point<3> (2.69642, 1.1866, 2.03), edgepoints.Get(1)) < 1e-6 ||
+      Dist2 (Point<3> (2.69642, 1.1866, 2.03), edgepoints.Last()) < 1e-6;
+    */
+
+    if (debug)
+      {
+	//      (*testout) << "tubious edge !!!" << endl;
+	(*testout) << "s1, s2 = " << s1 << " - " << s2 << endl;
+      }
+
+    refedges.SetSize(0);
+    refedgesinv.SetSize(0);
+    hp = Center (edgepoints.Get(1), edgepoints.Get(2));
+    ProjectToEdge (geometry.GetSurface(s1), geometry.GetSurface(s2), hp);
+
+    geometry.GetSurface(s1) -> CalcGradient (hp, a1);
+    geometry.GetSurface(s2) -> CalcGradient (hp, a2);
+    t = Cross (a1, a2);
+    t.Normalize();
+    if (!pos) t *= -1;    
+  
+    (*testout) << "t = " << t << endl;
+
+    for (i = 0; i < geometry.GetNTopLevelObjects(); i++)
+      {
+	(*testout) << "layer = " << layer 
+		   << ", tlo-layer = " << geometry.GetTopLevelObject(i)->GetLayer() << endl;
+	if (geometry.GetTopLevelObject(i)->GetLayer() != layer) 
+	  continue;
+      
+	const Solid * sol = geometry.GetTopLevelObject(i)->GetSolid();
+	const Surface * surf = geometry.GetTopLevelObject(i)->GetSurface();
+
+	sol -> TangentialSolid (hp, locsol);
+	if (!locsol) continue;
+
+	BoxSphere<3> boxp (hp, hp);
+	boxp.Increase (1e-5);
+	boxp.CalcDiamCenter();
+      
+	ReducePrimitiveIterator rpi(boxp);
+	UnReducePrimitiveIterator urpi;
+      
+	((Solid*)locsol) -> IterateSolid (rpi);
+
+	locsol -> CalcSurfaceInverse ();
+      
+
+	if (!surf)
+	  {
+	    locsol -> GetSurfaceIndices (locsurfind);
+	  }
+	else
+	  {
+	    /*
+	      if (fabs (surf->CalcFunctionValue (hp)) < 1e-6)
+	      continue;
+	    */
+	    locsurfind.SetSize(1);
+	    locsurfind[0] = -1;
+	    for (j = 0; j < geometry.GetNSurf(); j++)
+	      if (geometry.GetSurface(j) == surf)
+		{
+		  locsurfind[0] = j;
+		  //		      geometry.GetSurfaceClassRepresentant(j);
+		  break;
+		}
+	  }
+
+	((Solid*)locsol) -> IterateSolid (urpi);
+
+      
+	if (debug)
+	  (*testout) << "edge of tlo " << i << ", has " << locsurfind.Size() << " faces." << endl;
+      
+
+	for (j = locsurfind.Size()-1; j >= 0; j--)
+	  if (fabs (geometry.GetSurface(locsurfind[j])
+		    ->CalcFunctionValue (hp) ) > 1e-6)
+	    locsurfind.DeleteElement(j+1);
+      
+	if (debug)
+	  (*testout) << locsurfind.Size() << " faces on hp" << endl;
+
+	for (j = 0; j < locsurfind.Size(); j++)
+	  {      
+	    int lsi = locsurfind[j];
+	    int rlsi = geometry.GetSurfaceClassRepresentant(lsi);
+	  
+	    Vec<3> rn;
+
+	    // n is outer normal to solid
+	    geometry.GetSurface(lsi) -> GetNormalVector (hp, n);
+	    if (geometry.GetSurface (lsi)->Inverse())
+	      n *= -1;
+	  
+	    if (fabs (t * n) > 1e-4) continue;
+	    if (debug)
+	      {
+		(*testout) << "face " << locsurfind.Get(j) << ", rep = " << rlsi 
+			   << " has (t*n) = " << (t*n) << endl;
+		(*testout) << "n = " << n << endl;
+	      }
+	  
+	    // rn is normal to class representant
+	    geometry.GetSurface(rlsi) -> GetNormalVector (hp, rn);
+	  
+	    int sameasref = ((n * rn) > 0);
+	  
+	    m = Cross (t, rn);
+	    m.Normalize();
+	  
+
+	    for (k = 1; k <= 2; k ++)
+	      {
+		bool edgeinv = (k == 2);
+	      
+		if (debug)
+		  {
+		    (*testout) << "onface(" << hp << ", " << m << ")= " 
+			       << locsol->OnFace (hp, m);
+		    (*testout) << " vec2in = "
+			       << locsol -> VectorIn2 (hp, m, n) << " and " 
+			       << locsol -> VectorIn2 (hp, m, -1 * n) << endl;
+		  }
+
+		//	      if (locsol -> OnFace (hp, m))
+		if (locsol -> VectorIn2 (hp, m, n) == 0 &&
+		    locsol -> VectorIn2 (hp, m, -1 * n) == 1)
+		  {
+		    hi = 0;
+		    for (l = 1; l <= refedges.Size(); l++)
+		      {
+			if (refedges.Get(l).si == rlsi &&
+			    refedgesinv.Get(l) == edgeinv)
+			  hi = l;
+		      }
+		  
+		    if (!hi)
+		      {
+			seg.si = rlsi;
+			seg.domin = -1;
+			seg.domout = -1;
+			seg.tlosurf = -1;
+			seg.surfnr1 = s1;
+			seg.surfnr2 = s2;
+			hi = refedges.Append (seg);
+			refedgesinv.Append (edgeinv);
+		      }
+		  
+		    if (!surf)
+		      {
+			if (sameasref)
+			  refedges.Elem(hi).domin = i;
+			else 
+			  refedges.Elem(hi).domout = i;
+		      }
+		    else
+		      refedges.Elem(hi).tlosurf = i;
+
+		    if (debug)
+		      (*testout) << "add ref seg:" 
+				 << "si = " << refedges.Get(hi).si
+				 << ", domin = " << refedges.Get(hi).domin
+				 << ", domout = " << refedges.Get(hi).domout
+				 << ", surfnr1/2 = " << refedges.Get(hi).surfnr1
+				 << ", " << refedges.Get(hi).surfnr2
+				 << ", inv = " << refedgesinv.Get(hi) 
+				 << ", refedgenr = " << hi
+				 << endl;
+		  }
+		m *= -1;
+	      } 
+	  }
+	delete locsol;          
+      }
+  }
+
+
+
+  void EdgeCalculation :: 
+  StoreEdge (const ARRAY<Segment> & refedges,
+	     const ARRAY<int> & refedgesinv,
+	     const ARRAY<Point<3> > & edgepoints,
+	     const ARRAY<double> & curvelength,
+	     int layer,
+	     Mesh & mesh)
+  {
+  
+    // Calculate optimal element-length
+    int i, j, k;
+    PointIndex pi;
+    int ne;
+
+    double len, corr, lam;
+    PointIndex thispi, lastpi;
+    Point<3> p, np;
+    Segment seg;
+
+
+    const Surface * surf1 = geometry.GetSurface (refedges.Get(1).surfnr1);
+    const Surface * surf2 = geometry.GetSurface (refedges.Get(1).surfnr2);
+
+    len = curvelength.Last();
+    ne = int (len + 0.5);
+    if (ne == 0) ne = 1;
+    if (Dist2 (edgepoints.Get(1), edgepoints.Last()) < 1e-8 && 
+	ne <= 6) 
+      ne = 6;
+    corr = len / ne;
+
+    // generate initial point
+    p = edgepoints.Get(1);
+    lastpi = -1;
+    for (pi = PointIndex::BASE; 
+	 pi < mesh.GetNP()+PointIndex::BASE; pi++)
+      if (Dist (mesh[pi], p) < 1e-6)
+	{
+	  lastpi = pi;
+	  break;
+	}
+
+    if (lastpi == -1)
+      lastpi = mesh.AddPoint (p, layer);
+
+  
+    j = 1;
+    for (i = 1; i <= ne; i++)
+      {
+	while (curvelength.Get(j) < i * corr && j < curvelength.Size()) j++;
+      
+	lam = (i * corr - curvelength.Get(j-1)) / 
+	  (curvelength.Get(j) - curvelength.Get(j-1));
+      
+	np(0) = (1-lam) * edgepoints.Get(j-1)(0) + lam * edgepoints.Get(j)(0);
+	np(1) = (1-lam) * edgepoints.Get(j-1)(1) + lam * edgepoints.Get(j)(1);
+	np(2) = (1-lam) * edgepoints.Get(j-1)(2) + lam * edgepoints.Get(j)(2);
+      
+      
+	thispi = -1;
+	if (i == ne)
+	  for (pi = PointIndex::BASE; 
+	       pi < mesh.GetNP()+PointIndex::BASE; pi++)
+	    if (Dist(mesh[pi], np) < 1e-6)
+	      thispi = pi;
+      
+	if (thispi == -1)
+	  {
+	    ProjectToEdge (surf1, surf2, np);
+	    thispi = mesh.AddPoint (np, layer);
+	  }
+
+	for (k = 1; k <= refedges.Size(); k++)
+	  {
+	    if (refedgesinv.Get(k))
+	      {
+		seg.p1 = lastpi;
+		seg.p2 = thispi;
+	      }
+	    else
+	      {
+		seg.p1 = thispi;
+		seg.p2 = lastpi;
+	      }
+	    seg.si = refedges.Get(k).si;
+	    seg.domin = refedges.Get(k).domin;
+	    seg.domout = refedges.Get(k).domout;
+	    seg.tlosurf = refedges.Get(k).tlosurf;
+	    seg.edgenr = refedges.Get(k).edgenr;
+	    seg.surfnr1 = refedges.Get(k).surfnr1;
+	    seg.surfnr2 = refedges.Get(k).surfnr2;
+	    seg.seginfo = 0;
+	    if (k == 1) seg.seginfo = (refedgesinv.Get(k)) ? 2 : 1;
+	    mesh.AddSegment (seg);
+	    //	  (*testout) << "add seg " << seg.p1 << "-" << seg.p2 << endl;
+	  
+	    double maxh = min2 (geometry.GetSurface(seg.surfnr1)->GetMaxH(),
+				geometry.GetSurface(seg.surfnr2)->GetMaxH());
+			      
+	    if (seg.domin != -1)
+	      {
+		const Solid * s1 = 
+		  geometry.GetTopLevelObject(seg.domin) -> GetSolid();
+		maxh = min2 (maxh, s1->GetMaxH());
+		maxh = min2 (maxh, geometry.GetTopLevelObject(seg.domin)->GetMaxH());
+		mesh.RestrictLocalH (p, maxh);
+		mesh.RestrictLocalH (np, maxh);
+	      }
+	    if (seg.domout != -1)
+	      {
+		const Solid * s1 = 
+		  geometry.GetTopLevelObject(seg.domout) -> GetSolid();
+		maxh = min2 (maxh, s1->GetMaxH());
+		maxh = min2 (maxh, geometry.GetTopLevelObject(seg.domout)->GetMaxH());
+		mesh.RestrictLocalH (p, maxh);
+		mesh.RestrictLocalH (np, maxh);
+	      }
+	    if (seg.tlosurf != -1)
+	      {
+		double hi = geometry.GetTopLevelObject(seg.tlosurf) -> GetMaxH();
+		maxh = min2 (maxh, hi);
+		mesh.RestrictLocalH (p, maxh);
+		mesh.RestrictLocalH (np, maxh);
+	      }	  
+	  }
+      
+	p = np;
+	lastpi = thispi;
+      }
+
+#ifdef DEVELOP
+    (*testout) << " eplast = " << lastpi << " = " << p << endl;
+#endif
+  }
+  
+
+
+
+
+
+  void EdgeCalculation :: 
+  StoreShortEdge (const ARRAY<Segment> & refedges,
+		  const ARRAY<int> & refedgesinv,
+		  const ARRAY<Point<3> > & edgepoints,
+		  const ARRAY<double> & curvelength,
+		  int layer,
+		  Mesh & mesh)
+  {
+  
+    // Calculate optimal element-length
+    int i, j, k;
+    PointIndex pi;
+    int ne;
+    Segment seg;
+
+    /*
+      double len, corr, lam;
+      int thispi, lastpi;
+      Point<3> p, np;
+
+
+      const Surface * surf1 = geometry.GetSurface (refedges.Get(1).surfnr1);
+      const Surface * surf2 = geometry.GetSurface (refedges.Get(1).surfnr2);
+
+      len = curvelength.Last();
+      ne = int (len + 0.5);
+      if (ne == 0) ne = 1;
+      if (Dist2 (edgepoints[1], edgepoints.Last()) < 1e-8 && 
+      ne <= 6) 
+      ne = 6;
+      corr = len / ne;
+    */
+
+    // generate initial point
+    Point<3> p = edgepoints[0];
+    PointIndex pi1 = -1;
+    for (pi = PointIndex::BASE; 
+	 pi < mesh.GetNP()+PointIndex::BASE; pi++)
+
+      if (Dist (mesh[pi], p) < 1e-6)
+	{
+	  pi1 = pi;
+	  break;
+	}
+
+    if (pi1 == -1) pi1 = mesh.AddPoint (p, layer);
+
+    p = edgepoints.Last();
+    PointIndex pi2 = -1;
+    for (pi = PointIndex::BASE; 
+	 pi < mesh.GetNP()+PointIndex::BASE; pi++)
+
+      if (Dist (mesh[pi], p) < 1e-6)
+	{
+	  pi2 = pi;
+	  break;
+	}
+    if (pi2==-1) pi2 = mesh.AddPoint (p, layer);
+
+    /*
+  
+    j = 1;
+    for (i = 1; i <= ne; i++)
+    {
+    while (curvelength[j] < i * corr && j < curvelength.Size()) j++;
+      
+    lam = (i * corr - curvelength[j-1]) / 
+    (curvelength[j] - curvelength[j-1]);
+      
+    np(0) = (1-lam) * edgepoints[j-1](0) + lam * edgepoints[j](0);
+    np(1) = (1-lam) * edgepoints[j-1](1) + lam * edgepoints[j](1);
+    np(2) = (1-lam) * edgepoints[j-1](2) + lam * edgepoints[j](2);
+      
+      
+    thispi = 0;
+    if (i == ne)
+    for (j = 1; j <= mesh.GetNP(); j++)
+    if (Dist(mesh.Point(j), np) < 1e-6)
+    thispi = j;
+      
+    if (!thispi)
+    {
+    ProjectToEdge (surf1, surf2, np);
+    thispi = mesh.AddPoint (np);
+    }
+    */
+  
+    for (k = 1; k <= refedges.Size(); k++)
+      {
+	if (refedgesinv.Get(k))
+	  {
+	    seg.p1 = pi1;
+	    seg.p2 = pi2;
+	  }
+	else
+	  {
+	    seg.p1 = pi2;
+	    seg.p2 = pi1;
+	  }
+
+	seg.si = refedges.Get(k).si;
+	seg.domin = refedges.Get(k).domin;
+	seg.domout = refedges.Get(k).domout;
+	seg.tlosurf = refedges.Get(k).tlosurf;
+	seg.edgenr = refedges.Get(k).edgenr;
+	seg.surfnr1 = refedges.Get(k).surfnr1;
+	seg.surfnr2 = refedges.Get(k).surfnr2;
+	seg.seginfo = 0;
+	if (k == 1) seg.seginfo = (refedgesinv.Get(k)) ? 2 : 1;
+	mesh.AddSegment (seg);
+	//	  (*testout) << "add seg " << seg.p1 << "-" << seg.p2 << endl;
+      }
+  }
+  
+
+
+
+
+
+
+  void EdgeCalculation :: 
+  CopyEdge (const ARRAY<Segment> & refedges,
+	    const ARRAY<int> & refedgesinv,
+	    int copyfromedge, 
+	    const Point<3> & fromstart, const Point<3> & fromend,
+	    const Point<3> & tostart, const Point<3> & toend,
+	    int copyedgeidentification, 
+	    int layer,
+	    Mesh & mesh)
+  {
+    int i, j, k;
+    PointIndex pi;
+
+    // copy start and end points
+    for (i = 1; i <= 2; i++)
+      {
+	Point<3> fromp =
+	  (i == 1) ? fromstart : fromend;
+	Point<3> top =
+	  (i == 1) ? tostart : toend;
+      
+	PointIndex frompi = -1;
+	PointIndex topi = -1;
+	for (pi = PointIndex::BASE; 
+	     pi < mesh.GetNP()+PointIndex::BASE; pi++)
+	  {
+	    if (Dist2 (mesh[pi], fromp) <= 1e-16)
+	      frompi = pi;
+	    if (Dist2 (mesh[pi], top) <= 1e-16)
+	      topi = pi;
+	  }
+
+	if (topi == -1)
+	  topi = mesh.AddPoint (top, layer);
+
+	const Identification & csi = 
+	  (*geometry.identifications.Get(copyedgeidentification));
+
+	if (csi.Identifyable (mesh[frompi], mesh[topi]))
+	  mesh.GetIdentifications().Add(frompi, topi, copyedgeidentification);
+	else if (csi.Identifyable (mesh[topi], mesh[frompi]))
+	  mesh.GetIdentifications().Add(topi, frompi, copyedgeidentification);
+	else
+	  {
+	    cerr << "edgeflw.cpp: should identify, but cannot";
+	    exit(1);
+	  }
+	/*
+	  (*testout) << "Add Identification from CopyEdge, p1 = " 
+	  << mesh[PointIndex(frompi)] << ", p2 = " 
+	  << mesh[PointIndex(topi)] << endl;
+
+	  mesh.GetIdentifications().Add(frompi, topi, copyedgeidentification);
+	*/
+      }
+
+    int oldns = mesh.GetNSeg();
+    for (i = 1; i <= oldns; i++)
+      {
+	// real copy, since array might be reallocated !!
+	const Segment oldseg = mesh.LineSegment(i);
+	if (oldseg.edgenr != copyfromedge)
+	  continue;
+	if (oldseg.seginfo == 0)
+	  continue;
+
+	int pi1 = oldseg.p1;
+	int pi2 = oldseg.p2;
+
+	int npi1 = geometry.identifications.Get(copyedgeidentification)
+	  -> GetIdentifiedPoint (mesh, pi1);
+	int npi2 = geometry.identifications.Get(copyedgeidentification)
+	  -> GetIdentifiedPoint (mesh, pi2);
+
+	Segment seg;
+
+	for (k = 1; k <= refedges.Size(); k++)
+	  {
+	    int inv = refedgesinv.Get(k);
+
+	    // other edge is inverse
+	    if (oldseg.seginfo == 1)
+	      inv = !inv;
+
+	    //	  (*testout) << "inv, now = " << inv << endl;
+
+	    if (inv)
+	      {
+		seg.p1 = npi1;
+		seg.p2 = npi2;
+	      }
+	    else
+	      {
+		seg.p1 = npi2;
+		seg.p2 = npi1;
+	      }
+	    seg.si = refedges.Get(k).si;
+	    seg.domin = refedges.Get(k).domin;
+	    seg.domout = refedges.Get(k).domout;
+	    seg.tlosurf = refedges.Get(k).tlosurf;
+	    seg.edgenr = refedges.Get(k).edgenr;
+	    seg.surfnr1 = refedges.Get(k).surfnr1;
+	    seg.surfnr2 = refedges.Get(k).surfnr2;
+	    seg.seginfo = 0;
+	    if (k == 1) seg.seginfo = refedgesinv.Get(k) ? 2 : 1;
+	    mesh.AddSegment (seg);
+	    //	  (*testout) << "copy seg " << seg.p1 << "-" << seg.p2 << endl;
+#ifdef DEVELOP
+
+	    (*testout) << "copy seg, face = " << seg.si << ": " 
+		       << " inv = " << inv << ", refinv = " << refedgesinv.Get(k)
+		       << mesh.Point(seg.p1) << ", " << mesh.Point(seg.p2) << endl;
+#endif
+
+	  }
+      
+      }   
+  }
+  
+
+
+
+
+
+
+  void EdgeCalculation :: 
+  FindClosedSurfaces (double h, Mesh & mesh)
+  {
+    // if there is no special point at a sphere, one has to add a segment pair
+  
+    int i, j; 
+    int nsol; 
+    int nsurf = geometry.GetNSurf();
+    int layer;
+
+    BitArray pointatsurface (nsurf);
+    Point<3> p1, p2;
+    Vec<3> nv, tv;
+    Solid * tansol;
+    ARRAY<int> tansurfind;
+    //  const Solid * sol;
+
+    nsol = geometry.GetNTopLevelObjects();
+
+
+    pointatsurface.Clear();
+  
+    /*
+      for (i = 1; i <= specpoints.Size(); i++)
+      {
+      int classrep;
+
+      classrep = geometry.GetSurfaceClassRepresentant (specpoints[i].s1);
+      pointatsurface.Set (classrep);
+      classrep = geometry.GetSurfaceClassRepresentant (specpoints[i].s2);
+      pointatsurface.Set (classrep);
+      //      pointatsurface.Set (specpoints[i].s1);
+      //      pointatsurface.Set (specpoints[i].s2);
+      }
+    */
+    for (i = 1; i <= mesh.GetNSeg(); i++)
+      {
+	const Segment & seg = mesh.LineSegment(i);
+	int classrep;
+
+#ifdef DEVELOP      
+	(*testout) << seg.surfnr1 << ", " << seg.surfnr2 << ", si = " << seg.si << endl;
+#endif
+	classrep = geometry.GetSurfaceClassRepresentant (seg.si);
+
+	pointatsurface.Set (classrep);
+      }
+
+  
+    for (i = 0; i < nsurf; i++)
+      {
+	int classrep = geometry.GetSurfaceClassRepresentant (i);
+
+	if (!pointatsurface.Test(classrep))
+	  {
+	    const Surface * s = geometry.GetSurface(i);
+	    p1 = s -> GetSurfacePoint();
+	    s -> GetNormalVector (p1, nv);
+		    
+	    double hloc = 
+	      min2 (s->LocH (p1, 3, 1, h), mesh.GetH(p1));
+
+	    tv = nv.GetNormal ();
+	    tv *=  (hloc / tv.Length());
+	    p2 = p1 + tv;
+	    s->Project (p2);
+	  
+		    
+	    Segment seg1;
+	    seg1.si = i;
+	    seg1.domin = -1;
+	    seg1.domout = -1;
+
+	    Segment seg2;
+	    seg2.si = i;
+	    seg2.domin = -1;
+	    seg2.domout = -1;
+
+	    seg1.surfnr1 = i;
+	    seg2.surfnr1 = i;
+	    seg1.surfnr2 = i;
+	    seg2.surfnr2 = i;
+
+	    for (j = 0; j < nsol; j++)
+	      {
+		if (geometry.GetTopLevelObject(j)->GetSurface())
+		  continue;
+
+		const Solid * sol = geometry.GetTopLevelObject(j)->GetSolid();
+		sol -> TangentialSolid (p1, tansol);
+		layer = geometry.GetTopLevelObject(j)->GetLayer();
+
+		if (tansol)
+		  {
+		    tansol -> GetSurfaceIndices (tansurfind);
+		
+		    if (tansurfind.Size() == 1 && tansurfind.Get(1) == i)
+		      {
+			if (!tansol->VectorIn(p1, nv))
+			  {
+			    seg1.domin = j;
+			    seg2.domin = j;
+			    seg1.tlosurf = j;
+			    seg2.tlosurf = j;
+			  }
+			else
+			  {
+			    seg1.domout = j;
+			    seg2.domout = j;
+			    seg1.tlosurf = j;
+			    seg2.tlosurf = j;
+			  }
+			//        seg.s2 = i;
+			//        seg.invs1 = surfaces[i] -> Inverse();
+			//        seg.invs2 = ! (surfaces[i] -> Inverse());
+		      }
+		    delete tansol;
+		  }
+	      }
+
+
+	    if (seg1.domin != -1 || seg1.domout != -1)
+	      {
+		mesh.AddPoint (p1, layer);
+		mesh.AddPoint (p2, layer);
+		seg1.p1 = mesh.GetNP()-1;
+		seg1.p2 = mesh.GetNP();
+		seg2.p2 = mesh.GetNP()-1;
+		seg2.p1 = mesh.GetNP();
+		seg1.geominfo[0].trignum = 1;
+		seg1.geominfo[1].trignum = 1;
+		seg2.geominfo[0].trignum = 1;
+		seg2.geominfo[1].trignum = 1;
+		mesh.AddSegment (seg1);
+		mesh.AddSegment (seg2);
+
+		PrintMessage (5, "Add line segment to smooth surface");
+
+#ifdef DEVELOP
+		(*testout) << "Add segment at smooth surface " << i;
+		if (i != classrep) (*testout) << ", classrep = " << classrep;
+		(*testout) << ": "
+			   << mesh.Point (mesh.GetNP()-1) << " - "
+			   << mesh.Point (mesh.GetNP()) << endl;
+#endif
+	      }
+	  }
+      }
+  }
+
+}
diff --git a/Netgen/libsrc/csg/edgeflw_new.cpp b/Netgen/libsrc/csg/edgeflw_new.cpp
new file mode 100644
index 0000000000..8aa9f0263b
--- /dev/null
+++ b/Netgen/libsrc/csg/edgeflw_new.cpp
@@ -0,0 +1,1553 @@
+#include <mystdlib.h>
+#include <meshing.hpp>
+#include <csg.hpp>
+
+#undef DEVELOP
+
+namespace netgen
+{
+
+  EdgeCalculation :: 
+  EdgeCalculation (const CSGeometry & ageometry,
+		   ARRAY<SpecialPoint> & aspecpoints)
+    : geometry(ageometry), specpoints(aspecpoints)
+  {
+    Box<3> bbox;
+    if (specpoints.Size() >= 1)
+      bbox.Set (specpoints[0].p);
+    for (int i = 1; i < specpoints.Size(); i++)
+      bbox.Add (specpoints[i].p);
+
+    searchtree = new Point3dTree (bbox.PMin(), bbox.PMax());
+    meshpoint_tree = new Point3dTree (bbox.PMin(), bbox.PMax());
+
+    for (int i = 0; i < specpoints.Size(); i++)
+      searchtree->Insert (specpoints[i].p, i);
+  }
+
+  EdgeCalculation :: ~EdgeCalculation()
+  {
+    delete searchtree;
+    delete meshpoint_tree;
+  }
+
+
+  void EdgeCalculation :: Calc(double h, Mesh & mesh)
+  {
+    (*testout) << "Find edges" << endl;
+    PrintMessage (1, "Find edges");
+    PushStatus ("Find edges");
+
+    CalcEdges1 (h, mesh);
+    SplitEqualOneSegEdges (mesh);
+    FindClosedSurfaces (h, mesh);
+    PrintMessage (3, cntedge, " edges found");
+
+    for (SegmentIndex si = 0; si < mesh.GetNSeg(); si++)
+      (*testout) << "seg " << si << " = " << mesh[si] << endl;
+    PopStatus ();
+  }
+
+
+
+
+
+  void EdgeCalculation :: CalcEdges1 (double h, Mesh & mesh)
+  {
+    ARRAY<int> hsp(specpoints.Size());
+    ARRAY<int> glob2hsp(specpoints.Size());
+    ARRAY<int> startpoints, endpoints;
+
+    int i, j, k, l, hi, pos, ep, ne;
+    int layer;
+
+    Vec<3> a1, a2, t, n, m;
+    Point<3> p, np, pnp, hp;
+
+    Segment seg;
+    int pi1, s1, s2;
+    int lastpi, thispi;
+
+    ARRAY<Point<3> > edgepoints;
+    ARRAY<double> curvelength;
+    int copyedge, copyfromedge, copyedgeidentification;
+
+    ARRAY<int> locsurfind, locind;
+
+    double len, corr, lam;
+    double steplen, cursteplen, loch, hd;
+
+    int checkedcopy = 0;
+
+    double size = geometry.MaxSize(); // globflags.GetNumFlag ("maxsize", 500);
+    double epspointdist2 = size * 1e-6; // globflags.GetNumFlag ("epspointdist", size * 1e-6);
+    epspointdist2 = sqr (epspointdist2);
+
+    
+
+    Solid * locsol;
+
+
+    // copy special points to work with
+    for (i = 0; i < specpoints.Size(); i++)
+      {
+	hsp[i] = i;
+	glob2hsp[i] = i;
+      }
+
+
+    cntedge = 0;
+
+    while (hsp.Size())
+      {
+	SetThreadPercent(100 - 100 * double (hsp.Size()) / specpoints.Size());
+
+	edgepoints.SetSize (0);
+	curvelength.SetSize (0);
+      
+
+	pi1 = 0;
+	copyedge = 0;
+	// identifyable point available ?
+
+	//      (*testout) << endl;
+
+	for (i = 1; i <= geometry.identifications.Size() && !pi1; i++)
+	  {
+	    for (j = checkedcopy+1; j <= startpoints.Size() && !pi1; j++)
+	      {
+
+		if (geometry.identifications.Get(i)->IdentifyableCandidate (specpoints[startpoints.Get(j)]))
+		
+		  {
+		    int pi1cand = 0;
+		    double mindist = 1e10;
+		  
+		    for (k = 1; k <= hsp.Size() && !pi1; k++)
+		      {
+#ifdef DEVELOP
+			(*testout) << "check kand = " << hsp.Get(k).p 
+				   << ", v = " << hsp.Get(k).v 
+				   << endl;		      
+#endif
+			if (geometry.identifications.Get(i)
+			    ->Identifyable(specpoints[startpoints.Get(j)], specpoints[hsp.Get(k)]) ||
+			    geometry.identifications.Get(i)
+			    ->Identifyable(specpoints[hsp.Get(k)], specpoints[startpoints.Get(j)]))
+			  {
+
+#ifdef DEVELOP
+			    (*testout) << "identifiable, dist = "
+				       << Dist (startpoints.Get(j).p, hsp.Get(k).p) << endl;
+#endif
+
+			    if (Dist (specpoints[startpoints.Get(j)].p, specpoints[hsp.Get(k)].p) < mindist)
+			      {
+				mindist = Dist (specpoints[startpoints.Get(j)].p, specpoints[hsp.Get(k)].p);
+				pi1cand = k;
+			      }
+			    /*
+			      pi1 = k;
+			      copyedge = 1;
+			      copyfromedge = j;
+			      copyedgeidentification = i;
+			  
+			      (*testout) << "copy edge startpoint from "
+			      << startpoints.Get(j).p << " - " 
+			      << startpoints.Get(j).v 
+			      << " to " 
+			      << hsp.Get(k).p << " - " << hsp.Get(k).v << endl;
+			    */
+			  }
+		      }
+
+		    if (pi1cand)
+		      {
+			pi1 = pi1cand;
+			copyedge = 1;
+			copyfromedge = j;
+			copyedgeidentification = i;
+#ifdef DEVELOP
+			(*testout) << "copy edge startpoint from "
+				   << startpoints.Get(j).p << " - " 
+				   << startpoints.Get(j).v 
+				   << " to " 
+				   << specpoints[hsp.Get(pi1)].p << " - " << hsp.Get(pi1).v << endl;
+#endif
+		      }
+		  }
+	      }
+	  }
+      
+      
+	// cannot copy from other ege ?
+	if (!pi1)
+	  checkedcopy = startpoints.Size();
+      
+	// unconditional special point available ?
+	if (!pi1)
+	  for (i = 1; i <= hsp.Size(); i++)
+	    if (specpoints[hsp.Get(i)].unconditional == 1)
+	      {
+		pi1 = i;
+		break;
+	      }
+ 
+     
+	if (!pi1)
+	  {
+	    // only unconditional points available, choose first
+	    pi1 = 1;
+	  }
+
+	layer = specpoints[hsp.Get(pi1)].GetLayer();
+      
+
+	if (!specpoints[hsp.Get(pi1)].unconditional)
+	  {
+	    specpoints[hsp.Elem(pi1)].unconditional = 1;
+	    for (i = 1; i <= hsp.Size(); i++)
+	      if (i != pi1 && 
+		  Dist (specpoints[hsp.Get(pi1)].p, specpoints[hsp.Get(i)].p) < 1e-8 &&
+		  (specpoints[hsp.Get(pi1)].v + specpoints[hsp.Get(i)].v).Length() < 1e-4)
+		{
+		  // opposite direction
+		  specpoints[hsp.Elem(i)].unconditional = 1;
+		}
+	  }
+
+	cntedge++;
+	startpoints.Append (hsp.Get(pi1));
+
+#ifdef DEVELOP
+	(*testout) << "edge nr " << cntedge << endl;
+	(*testout) << "start followedge: p1 = " << hsp.Get(pi1).p << ", v = " << hsp.Get(pi1).v << endl;
+#endif
+
+	FollowEdge (pi1, ep, pos, hsp, h, mesh,
+		    edgepoints, curvelength);
+
+
+	if (multithread.terminate)
+	  return;
+      
+	if (!ep)
+	  {
+	    // ignore starting point
+	    hsp.DeleteElement (pi1);
+	    cout << "yes, this happens" << endl;
+	    continue;
+	  }
+
+
+
+	endpoints.Append (hsp.Get(ep));
+
+
+	double elen = 0;
+	for (i = 1; i <= edgepoints.Size()-1; i++)
+	  elen += Dist (edgepoints.Get(i), edgepoints.Get(i+1));
+
+
+	int shortedge = 0;
+	for (i = 1; i <= geometry.identifications.Size(); i++)
+	  if (geometry.identifications.Get(i)->ShortEdge(specpoints[hsp.Get(pi1)], specpoints[hsp.Get(ep)]))
+	    shortedge = 1;
+	// (*testout) << "shortedge = " << shortedge << endl;
+
+
+	if (!shortedge)
+	  {
+	    mesh.RestrictLocalHLine (Point3d (specpoints[hsp.Get(pi1)].p), 
+				     Point3d (specpoints[hsp.Get(ep)].p), 
+				     elen / mparam.segmentsperedge);
+	  }
+      
+	s1 = specpoints[hsp.Get(pi1)].s1;
+	s2 = specpoints[hsp.Get(pi1)].s2;
+
+
+	// delete initial, terminal and conditional points
+
+#ifdef DEVELOP
+	(*testout) << "terminal point: p = " << hsp.Get(ep).p << ", v = " << hsp.Get(ep).v << endl;      
+#endif
+
+	searchtree -> DeleteElement (hsp.Get(ep));
+	searchtree -> DeleteElement (hsp.Get(pi1));
+
+	if (ep > pi1)
+	  {
+	    glob2hsp[hsp[ep-1]] = -1;
+	    glob2hsp[hsp.Last()] = ep-1;
+	    hsp.DeleteElement (ep);
+
+	    glob2hsp[hsp[pi1-1]] = -1;
+	    glob2hsp[hsp.Last()] = pi1-1;
+	    hsp.DeleteElement (pi1);
+	  }
+	else
+	  {
+	    glob2hsp[hsp[pi1-1]] = -1;
+	    glob2hsp[hsp.Last()] = pi1-1;
+	    hsp.DeleteElement (pi1);
+
+	    glob2hsp[hsp[ep-1]] = -1;
+	    glob2hsp[hsp.Last()] = ep-1;
+	    hsp.DeleteElement (ep);
+	  }
+
+
+	for (j = 1; j <= edgepoints.Size()-1; j++)
+	  {
+	    p = edgepoints.Get(j);
+	    np = Center (p, edgepoints.Get(j+1));
+	    hd = Dist (p, np);
+ 
+
+	    Box<3> boxp (np - (1.2 * hd) * Vec<3> (1, 1, 1),
+			 np + (1.2 * hd) * Vec<3> (1, 1, 1));
+	    searchtree -> GetIntersecting (boxp.PMin(), boxp.PMax(), locind);	    
+
+	    for (i = 0; i < locind.Size(); i++)
+	      {
+		if ( specpoints[locind[i]].HasSurfaces (s1, s2) &&
+		     specpoints[locind[i]].unconditional == 0)
+		  {
+		    searchtree -> DeleteElement (locind[i]);
+
+		    int li = glob2hsp[locind[i]];
+		    glob2hsp[locind[i]] = -1;
+		    glob2hsp[hsp.Last()] = li;
+		    hsp.Delete (li);
+		  }
+	      }
+
+
+	    /*
+	    for (i = 1; i <= hsp.Size(); i++)
+	      if ( specpoints[hsp.Get(i)].HasSurfaces (s1, s2) &&
+		   specpoints[hsp.Get(i)].unconditional == 0 &&
+		   Dist2 (np, specpoints[hsp.Get(i)].p) < 1.2 * hd)
+		{
+		  searchtree -> DeleteElement (hsp.Get(i)+1);
+		  hsp.DeleteElement (i);
+		  i--;
+		}
+	    */
+	  }
+
+      
+	ARRAY<Segment> refedges;
+	ARRAY<int> refedgesinv;
+      
+
+	AnalyzeEdge (s1, s2, pos, layer,
+		     edgepoints,
+		     refedges, refedgesinv);
+
+	for (i = 1; i <= refedges.Size(); i++)
+	  refedges.Elem(i).edgenr = cntedge;
+
+
+#ifdef DEVELOP
+	(*testout) << "edge " << cntedge << endl
+		   << "startp: " << startpoints.Last().p 
+		   << ", v = " << startpoints.Last().v << endl
+		   << "copy = " << copyedge << endl
+		   << refedges.Size() << " refedges: ";
+	for (i = 1; i <= refedges.Size(); i++)
+	  (*testout) << " " << refedges.Get(i).si;
+	(*testout) << endl;
+	(*testout) << "inv[1] = " << refedgesinv.Get(1) << endl;
+#endif
+      
+	if (!copyedge)
+	  {
+	    int oldnseg = mesh.GetNSeg();
+
+	    if (!shortedge)
+	      StoreEdge (refedges, refedgesinv, 
+			 edgepoints, curvelength, layer, mesh);
+	    else
+	      StoreShortEdge (refedges, refedgesinv, 
+			      edgepoints, curvelength, layer, mesh);
+
+
+	    /*
+	      for (i = oldnseg+1; i <= mesh.GetNSeg(); i++)
+	      for (j = 1; j <= oldnseg; j++)
+	      {
+	      const Point<3> & l1p1 = mesh.Point (mesh.LineSegment(i).p1);
+	      const Point<3> & l1p2 = mesh.Point (mesh.LineSegment(i).p2);
+	      const Point<3> & l2p1 = mesh.Point (mesh.LineSegment(j).p1);
+	      const Point<3> & l2p2 = mesh.Point (mesh.LineSegment(j).p2);
+	      Vec<3> vl1(l1p1, l1p2);
+	      for (double lamk = 0; lamk <= 1; lamk += 0.1)
+	      {
+	      Point<3> l2p = l1p1 + lamk * vl1;
+	      double dist = sqrt (MinDistLP2 (l2p1, l2p2, l2p));
+	      if (dist > 1e-12)
+	      mesh.RestrictLocalH (l2p, 3*dist);
+	      }
+	      }
+	    */
+	  }
+	else
+	  {
+	    CopyEdge (refedges, refedgesinv,
+		      copyfromedge, 
+		      specpoints[startpoints.Get(copyfromedge)].p,
+		      specpoints[endpoints.Get(copyfromedge)].p,
+		      edgepoints.Get(1), edgepoints.Last(),
+		      copyedgeidentification, 
+		      layer,
+		      mesh);
+	  }
+
+      }
+  }
+
+
+
+  /*
+    If two or more edges share the same initial and end-points,
+    then they need at least two segments 
+  */
+  void EdgeCalculation ::
+  SplitEqualOneSegEdges (Mesh & mesh)
+  {
+    int i, j;
+    SegmentIndex si;
+    PointIndex pi;
+
+    ARRAY<int> osedges(cntedge);
+    INDEX_2_HASHTABLE<int> osedgesht (cntedge+1);
+
+    osedges = 2;
+
+    // count segments on edges
+    for (si = 0; si < mesh.GetNSeg(); si++)
+      {
+	const Segment & seg = mesh[si];
+	if (seg.seginfo && seg.edgenr >= 1 && seg.edgenr <= cntedge)
+	  osedges.Elem(seg.edgenr)--;
+      }
+
+    (*testout) << "osedges  = " << osedges << endl;
+
+    // flag one segment edges
+    for (i = 0; i < cntedge; i++)
+      osedges[i] = (osedges[i] > 0) ? 1 : 0;
+
+    (*testout) << "osedges, now  = " << osedges << endl;
+
+    for (si = 0; si < mesh.GetNSeg(); si++)
+      {
+	const Segment & seg = mesh[si];
+	if (seg.seginfo && seg.edgenr >= 1 && seg.edgenr <= cntedge)
+	  {
+	    if (osedges.Get(seg.edgenr))
+	      {
+		INDEX_2 i2(seg.p1, seg.p2);
+		i2.Sort ();
+		if (osedgesht.Used (i2))
+		  osedgesht.Set (i2, 2);
+		else
+		  osedgesht.Set (i2, 1);
+	      }
+	  }
+      }
+
+
+    // one edge 1 segment, other 2 segments 
+    // yes, it happens !
+  
+    for (i = 1; i <= osedgesht.GetNBags(); i++)
+      for (j = 1; j <= osedgesht.GetBagSize(i); j++)
+	{
+	  INDEX_2 i2; 
+	  int val;
+	  osedgesht.GetData (i, j, i2, val);
+
+	  const Point<3> & p1 = mesh[PointIndex(i2.I1())];
+	  const Point<3> & p2 = mesh[PointIndex(i2.I2())];
+	  Vec<3> v = p2 - p1;
+	  double vlen = v.Length();
+	  v /= vlen;
+	  for (pi = PointIndex::BASE; 
+	       pi < mesh.GetNP()+PointIndex::BASE; pi++)
+
+	    if (pi != i2.I1() && pi != i2.I2())
+	      {
+		const Point<3> & p = mesh[pi];
+		Vec<3> v2 = p - p1;
+		double lam = (v2 * v);
+		if (lam > 0 && lam < vlen)
+		  {
+		    Point<3> hp = p1 + lam * v;
+		    if (Dist (p, hp) < 1e-4 * vlen)
+		      {
+			PrintSysError ("Point on edge !!!");
+			cout << "seg: " << i2 << ", p = " << pi << endl;
+			osedgesht.Set (i2, 2);		      
+		      }
+		  }
+	      }
+	}
+
+
+    // insert new points
+    osedges = -1;
+
+    int nseg = mesh.GetNSeg();
+    for (si = 0; si < nseg; si++)
+      {
+	const Segment & seg = mesh[si];
+	if (seg.seginfo && seg.edgenr >= 1 && seg.edgenr <= cntedge)
+	  {
+	    INDEX_2 i2(seg.p1, seg.p2);
+	    i2.Sort ();
+	    if (osedgesht.Used (i2) &&
+		osedgesht.Get (i2) == 2 &&
+		osedges.Elem(seg.edgenr) == -1)
+	      {
+		Point<3> newp = Center (mesh[PointIndex(seg.p1)],
+					mesh[PointIndex(seg.p2)]);
+
+		ProjectToEdge (geometry.GetSurface(seg.surfnr1), 
+			       geometry.GetSurface(seg.surfnr2), 
+			       newp);
+
+		osedges.Elem(seg.edgenr) = 
+		  mesh.AddPoint (newp, mesh[PointIndex(seg.p1)].GetLayer());
+		meshpoint_tree -> Insert (newp, osedges.Elem(seg.edgenr));
+	      }
+	  }
+      }
+
+
+    for (i = 1; i <= nseg; i++)
+      {
+	Segment & seg = mesh.LineSegment (i);
+	if (seg.edgenr >= 1 && seg.edgenr <= cntedge)
+	  {
+	    if (osedges.Get(seg.edgenr) != -1)
+	      {
+		Segment newseg = seg;
+		newseg.p1 = osedges.Get(seg.edgenr);
+		seg.p2 = osedges.Get(seg.edgenr);
+		mesh.AddSegment (newseg);
+	      }
+	  }
+      }
+
+  }
+
+
+
+  void EdgeCalculation :: 
+  FollowEdge (int pi1, int & ep, int & pos,
+	      const ARRAY<int> & hsp,
+	      double h, const Mesh & mesh,
+	      ARRAY<Point<3> > & edgepoints,
+	      ARRAY<double> & curvelength)
+  {
+    int i, j, s1, s2;
+    double len, steplen, cursteplen, loch;
+    Point<3> p, np, pnp;
+    Vec<3> a1, a2, t;
+
+    ARRAY<int> locind;
+
+    double size = geometry.MaxSize();  
+    double epspointdist2 = size * 1e-6;
+    epspointdist2 = sqr (epspointdist2);
+    int uselocalh = mparam.uselocalh;
+
+
+    s1 = specpoints[hsp.Get(pi1)].s1;
+    s2 = specpoints[hsp.Get(pi1)].s2;
+  
+    p = specpoints[hsp.Get(pi1)].p;
+    geometry.GetSurface(s1) -> CalcGradient (p, a1);
+    geometry.GetSurface(s2) -> CalcGradient (p, a2);
+
+    t = Cross (a1, a2);
+    t.Normalize();
+
+    pos = (specpoints[hsp.Get(pi1)].v * t) > 0;
+    if (!pos) t *= -1;
+
+  
+    edgepoints.Append (p);
+    curvelength.Append (0);
+    len = 0;
+
+    loch = min2 (geometry.GetSurface(s1) -> LocH (p, 3, 1, h), 
+		 geometry.GetSurface(s2) -> LocH (p, 3, 1, h));
+  
+  
+  
+    if (uselocalh)
+      {
+	double lh = mesh.GetH(p);
+	if (lh < loch)
+	  loch = lh;
+      }
+
+    steplen = 0.1 * loch;
+  
+    do
+      {
+	if (multithread.terminate)
+	  return;
+      
+	if (fabs (p(0)) + fabs (p(1)) + fabs (p(2)) > 10000)
+	  {
+	    ep = 0;
+	    PrintWarning ("Give up line");
+	    break;
+	  }
+
+	if (steplen > 0.1 * loch) steplen = 0.1 * loch;
+      
+	steplen *= 2;
+	do
+	  {
+	    steplen *= 0.5;
+	    np = p + steplen * t;
+	    pnp = np;
+	    ProjectToEdge (geometry.GetSurface(s1), 
+			   geometry.GetSurface(s2), pnp);
+	  }
+	while (Dist (np, pnp) > 0.1 * steplen);
+      
+	cursteplen = steplen;
+	if (Dist (np, pnp) < 0.01 * steplen) steplen *= 2;
+      
+ 
+	np = pnp;
+      
+#ifdef MYGRAPH
+	if (silentflag <= 2)
+	  {
+	    MyLine3D (p, np, rot);
+	    MyDraw ();
+	  }
+#endif      
+
+	ep = 0;
+      
+	double hvtmin = 1.5 * cursteplen;
+      
+	Box<3> boxp (p - (2 * cursteplen) * Vec<3> (1, 1, 1),
+		     p + (2 * cursteplen) * Vec<3> (1, 1, 1));
+
+	searchtree -> GetIntersecting (boxp.PMin(), boxp.PMax(), locind);
+	
+	for (i = 0; i < locind.Size(); i++)
+	  {
+	    Vec<3> hv = specpoints[locind[i]].p - p;
+	    if (hv.Length2() > 9 * cursteplen * cursteplen)
+	      continue;
+
+	    double hvt = hv * t;
+	    hv -= hvt * t;
+	  
+	    if (hv.Length() < 0.2 * cursteplen &&
+		hvt > 0 && 
+		//		  hvt < 1.5 * cursteplen &&
+		hvt < hvtmin && 
+		specpoints[locind[i]].unconditional == 1 &&
+		(specpoints[locind[i]].v + t).Length() < 0.4  ) 
+	      {
+		Point<3> hep = specpoints[locind[i]].p;
+		ProjectToEdge (geometry.GetSurface(s1), 
+			       geometry.GetSurface(s2), hep);            
+	      
+	      
+		if (Dist2 (hep, specpoints[locind[i]].p) < epspointdist2 )
+		  {
+		    geometry.GetSurface(s1) -> CalcGradient (hep, a1);
+		    geometry.GetSurface(s2) -> CalcGradient (hep, a2);
+		    Vec<3> ept = Cross (a1, a2);
+		    ept /= ept.Length();
+		    if (!pos) ept *= -1;
+		  
+		    if ( (specpoints[locind[i]].v + ept).Length() < 1e-4 )
+		      {
+			np = specpoints[locind[i]].p;
+
+			for (int jj = 0; jj < hsp.Size(); jj++)
+			  if (hsp[jj] == locind[i])
+			    ep = jj+1;
+
+			if (!ep) 
+			  cerr << "endpoint not found" << endl;
+			  //			ep = i;
+			hvtmin = hvt;
+			//			  break;
+		      }
+		  }
+	      }
+	  }
+
+
+
+
+	/*
+	for (i = 1; i <= hsp.Size(); i++)
+	  {
+	    if (!boxp.IsIn (specpoints[hsp.Get(i)].p))
+	      continue;
+	  
+	    Vec<3> hv = specpoints[hsp.Get(i)].p - p;
+	    if (hv.Length2() > 9 * cursteplen * cursteplen)
+	      continue;
+
+	    double hvt = hv * t;
+	    hv -= hvt * t;
+	  
+	    if (hv.Length() < 0.2 * cursteplen &&
+		hvt > 0 && 
+		//		  hvt < 1.5 * cursteplen &&
+		hvt < hvtmin && 
+		specpoints[hsp.Get(i)].unconditional == 1 &&
+		(specpoints[hsp.Get(i)].v + t).Length() < 0.4  ) 
+	      {
+		Point<3> hep = specpoints[hsp.Get(i)].p;
+		ProjectToEdge (geometry.GetSurface(s1), 
+			       geometry.GetSurface(s2), hep);            
+	      
+	      
+		if (Dist2 (hep, specpoints[hsp.Get(i)].p) < epspointdist2 )
+		  {
+		    geometry.GetSurface(s1) -> CalcGradient (hep, a1);
+		    geometry.GetSurface(s2) -> CalcGradient (hep, a2);
+		    Vec<3> ept = Cross (a1, a2);
+		    ept /= ept.Length();
+		    if (!pos) ept *= -1;
+		  
+		    if ( (specpoints[hsp.Get(i)].v + ept).Length() < 1e-4 )
+		      {
+			np = specpoints[hsp.Get(i)].p;
+			ep = i;
+			hvtmin = hvt;
+			//			  break;
+		      }
+		  }
+	      }
+	  }
+	*/
+
+	loch = min2 (geometry.GetSurface(s1) -> LocH (np, 3, 1, h), 
+		     geometry.GetSurface(s2) -> LocH (np, 3, 1, h));
+
+	if (uselocalh)
+	  {
+	    double lh = mesh.GetH(np);
+	    if (lh < loch)
+	      loch = lh;
+	  }
+      
+      
+	len += Dist (p, np) / loch;
+	edgepoints.Append (np);
+	curvelength.Append (len);
+      
+	p = np;
+      
+	geometry.GetSurface(s1) -> CalcGradient (p, a1);
+	geometry.GetSurface(s2) -> CalcGradient (p, a2);
+	t = Cross (a1, a2);
+	t.Normalize();
+	if (!pos) t *= -1;
+      }
+    while (! ep);
+  }
+
+
+
+
+
+
+
+  void EdgeCalculation :: 
+  AnalyzeEdge (int s1, int s2, int pos, int layer,
+	       const ARRAY<Point<3> > & edgepoints,
+	       ARRAY<Segment> & refedges,
+	       ARRAY<int> & refedgesinv)
+  {
+    int i, j, k, l;
+    int hi;
+    Point<3> hp;
+    Vec<3> t, a1, a2, m, n;
+    Segment seg;
+    Solid * locsol;
+    ARRAY<int> locsurfind;
+
+    /*
+      int pi1 = 0, pi2 = 0;
+      extern Mesh * mesh;
+      for (i = 1; i <= mesh->GetNP(); i++)
+      {
+      if (Dist2 (edgepoints.Get(1), mesh->Point(i)) < 1e-12)
+      pi1 = i;
+      if (Dist2 (edgepoints.Last(), mesh->Point(i)) < 1e-12)
+      pi2 = i;
+      }
+      (*testout) << "Analyze edge: " << pi1 << " - " << pi2 << ", pts = " << edgepoints.Size() << endl;
+      (*testout) << "p1 = " << edgepoints.Get(1) << " pl = " << edgepoints.Last() << endl;
+    */
+    int debug = 0;
+    /*
+      Dist2 (Point<3> (2.69642, 1.1866, 2.03), edgepoints.Get(1)) < 1e-6 ||
+      Dist2 (Point<3> (2.69642, 1.1866, 2.03), edgepoints.Last()) < 1e-6;
+    */
+
+    if (debug)
+      {
+	//      (*testout) << "tubious edge !!!" << endl;
+	(*testout) << "s1, s2 = " << s1 << " - " << s2 << endl;
+      }
+
+    refedges.SetSize(0);
+    refedgesinv.SetSize(0);
+    hp = Center (edgepoints.Get(1), edgepoints.Get(2));
+    ProjectToEdge (geometry.GetSurface(s1), geometry.GetSurface(s2), hp);
+
+    geometry.GetSurface(s1) -> CalcGradient (hp, a1);
+    geometry.GetSurface(s2) -> CalcGradient (hp, a2);
+    t = Cross (a1, a2);
+    t.Normalize();
+    if (!pos) t *= -1;    
+  
+    (*testout) << "t = " << t << endl;
+
+    for (i = 0; i < geometry.GetNTopLevelObjects(); i++)
+      {
+	(*testout) << "layer = " << layer 
+		   << ", tlo-layer = " << geometry.GetTopLevelObject(i)->GetLayer() << endl;
+	if (geometry.GetTopLevelObject(i)->GetLayer() != layer) 
+	  continue;
+      
+	const Solid * sol = geometry.GetTopLevelObject(i)->GetSolid();
+	const Surface * surf = geometry.GetTopLevelObject(i)->GetSurface();
+
+	sol -> TangentialSolid (hp, locsol);
+	if (!locsol) continue;
+
+	BoxSphere<3> boxp (hp, hp);
+	boxp.Increase (1e-5);
+	boxp.CalcDiamCenter();
+      
+	ReducePrimitiveIterator rpi(boxp);
+	UnReducePrimitiveIterator urpi;
+      
+	((Solid*)locsol) -> IterateSolid (rpi);
+
+	locsol -> CalcSurfaceInverse ();
+      
+
+	if (!surf)
+	  {
+	    locsol -> GetSurfaceIndices (locsurfind);
+	  }
+	else
+	  {
+	    /*
+	      if (fabs (surf->CalcFunctionValue (hp)) < 1e-6)
+	      continue;
+	    */
+	    locsurfind.SetSize(1);
+	    locsurfind[0] = -1;
+	    for (j = 0; j < geometry.GetNSurf(); j++)
+	      if (geometry.GetSurface(j) == surf)
+		{
+		  locsurfind[0] = j;
+		  //		      geometry.GetSurfaceClassRepresentant(j);
+		  break;
+		}
+	  }
+
+	((Solid*)locsol) -> IterateSolid (urpi);
+
+      
+	if (debug)
+	  (*testout) << "edge of tlo " << i << ", has " << locsurfind.Size() << " faces." << endl;
+      
+
+	for (j = locsurfind.Size()-1; j >= 0; j--)
+	  if (fabs (geometry.GetSurface(locsurfind[j])
+		    ->CalcFunctionValue (hp) ) > 1e-6)
+	    locsurfind.DeleteElement(j+1);
+      
+	if (debug)
+	  (*testout) << locsurfind.Size() << " faces on hp" << endl;
+
+	for (j = 0; j < locsurfind.Size(); j++)
+	  {      
+	    int lsi = locsurfind[j];
+	    int rlsi = geometry.GetSurfaceClassRepresentant(lsi);
+	  
+	    Vec<3> rn;
+
+	    // n is outer normal to solid
+	    geometry.GetSurface(lsi) -> GetNormalVector (hp, n);
+	    if (geometry.GetSurface (lsi)->Inverse())
+	      n *= -1;
+	  
+	    if (fabs (t * n) > 1e-4) continue;
+	    if (debug)
+	      {
+		(*testout) << "face " << locsurfind.Get(j) << ", rep = " << rlsi 
+			   << " has (t*n) = " << (t*n) << endl;
+		(*testout) << "n = " << n << endl;
+	      }
+	  
+	    // rn is normal to class representant
+	    geometry.GetSurface(rlsi) -> GetNormalVector (hp, rn);
+	  
+	    int sameasref = ((n * rn) > 0);
+	  
+	    m = Cross (t, rn);
+	    m.Normalize();
+	  
+
+	    for (k = 1; k <= 2; k ++)
+	      {
+		bool edgeinv = (k == 2);
+	      
+		if (debug)
+		  {
+		    (*testout) << "onface(" << hp << ", " << m << ")= " 
+			       << locsol->OnFace (hp, m);
+		    (*testout) << " vec2in = "
+			       << locsol -> VectorIn2 (hp, m, n) << " and " 
+			       << locsol -> VectorIn2 (hp, m, -1 * n) << endl;
+		  }
+
+		//	      if (locsol -> OnFace (hp, m))
+		if (locsol -> VectorIn2 (hp, m, n) == 0 &&
+		    locsol -> VectorIn2 (hp, m, -1 * n) == 1)
+		  {
+		    hi = 0;
+		    for (l = 1; l <= refedges.Size(); l++)
+		      {
+			if (refedges.Get(l).si == rlsi &&
+			    refedgesinv.Get(l) == edgeinv)
+			  hi = l;
+		      }
+		  
+		    if (!hi)
+		      {
+			seg.si = rlsi;
+			seg.domin = -1;
+			seg.domout = -1;
+			seg.tlosurf = -1;
+			seg.surfnr1 = s1;
+			seg.surfnr2 = s2;
+			hi = refedges.Append (seg);
+			refedgesinv.Append (edgeinv);
+		      }
+		  
+		    if (!surf)
+		      {
+			if (sameasref)
+			  refedges.Elem(hi).domin = i;
+			else 
+			  refedges.Elem(hi).domout = i;
+		      }
+		    else
+		      refedges.Elem(hi).tlosurf = i;
+
+		    if (debug)
+		      (*testout) << "add ref seg:" 
+				 << "si = " << refedges.Get(hi).si
+				 << ", domin = " << refedges.Get(hi).domin
+				 << ", domout = " << refedges.Get(hi).domout
+				 << ", surfnr1/2 = " << refedges.Get(hi).surfnr1
+				 << ", " << refedges.Get(hi).surfnr2
+				 << ", inv = " << refedgesinv.Get(hi) 
+				 << ", refedgenr = " << hi
+				 << endl;
+		  }
+		m *= -1;
+	      } 
+	  }
+	delete locsol;          
+      }
+  }
+
+
+
+  void EdgeCalculation :: 
+  StoreEdge (const ARRAY<Segment> & refedges,
+	     const ARRAY<int> & refedgesinv,
+	     const ARRAY<Point<3> > & edgepoints,
+	     const ARRAY<double> & curvelength,
+	     int layer,
+	     Mesh & mesh)
+  {
+  
+    // Calculate optimal element-length
+    int i, j, k;
+    PointIndex pi;
+    int ne;
+
+    double len, corr, lam;
+    PointIndex thispi, lastpi;
+    Point<3> p, np;
+    Segment seg;
+
+
+    const Surface * surf1 = geometry.GetSurface (refedges.Get(1).surfnr1);
+    const Surface * surf2 = geometry.GetSurface (refedges.Get(1).surfnr2);
+
+    len = curvelength.Last();
+    ne = int (len + 0.5);
+    if (ne == 0) ne = 1;
+    if (Dist2 (edgepoints.Get(1), edgepoints.Last()) < 1e-8 && 
+	ne <= 6) 
+      ne = 6;
+    corr = len / ne;
+
+    // generate initial point
+    p = edgepoints.Get(1);
+    lastpi = -1;
+
+    /*
+    for (pi = PointIndex::BASE; 
+	 pi < mesh.GetNP()+PointIndex::BASE; pi++)
+      if (Dist (mesh[pi], p) < 1e-6)
+	{
+	  lastpi = pi;
+	  break;
+	}
+    */
+    ARRAY<int> locsearch;
+    meshpoint_tree -> GetIntersecting (p-Vec<3> (1e-6, 1e-6, 1e-6),
+				       p+Vec<3> (1e-6, 1e-6, 1e-6), locsearch);
+    if (locsearch.Size())
+      lastpi = locsearch[0];
+				       
+
+
+    if (lastpi == -1)
+      {
+	lastpi = mesh.AddPoint (p, layer);
+	meshpoint_tree -> Insert (p, lastpi); 
+      }
+  
+    j = 1;
+    for (i = 1; i <= ne; i++)
+      {
+	while (curvelength.Get(j) < i * corr && j < curvelength.Size()) j++;
+      
+	lam = (i * corr - curvelength.Get(j-1)) / 
+	  (curvelength.Get(j) - curvelength.Get(j-1));
+      
+	np(0) = (1-lam) * edgepoints.Get(j-1)(0) + lam * edgepoints.Get(j)(0);
+	np(1) = (1-lam) * edgepoints.Get(j-1)(1) + lam * edgepoints.Get(j)(1);
+	np(2) = (1-lam) * edgepoints.Get(j-1)(2) + lam * edgepoints.Get(j)(2);
+      
+      
+	thispi = -1;
+	if (i == ne)
+	  {
+	    /*
+	  for (pi = PointIndex::BASE; 
+	       pi < mesh.GetNP()+PointIndex::BASE; pi++)
+	    if (Dist(mesh[pi], np) < 1e-6)
+	      thispi = pi;
+	    */
+	    
+	    meshpoint_tree -> GetIntersecting (np-Vec<3> (1e-6, 1e-6, 1e-6),
+					       np+Vec<3> (1e-6, 1e-6, 1e-6), locsearch);
+	    if (locsearch.Size())
+	      thispi = locsearch[0];
+	  }
+
+	if (thispi == -1)
+	  {
+	    ProjectToEdge (surf1, surf2, np);
+	    thispi = mesh.AddPoint (np, layer);
+	    meshpoint_tree -> Insert (np, thispi);
+	  }
+
+	for (k = 1; k <= refedges.Size(); k++)
+	  {
+	    if (refedgesinv.Get(k))
+	      {
+		seg.p1 = lastpi;
+		seg.p2 = thispi;
+	      }
+	    else
+	      {
+		seg.p1 = thispi;
+		seg.p2 = lastpi;
+	      }
+	    seg.si = refedges.Get(k).si;
+	    seg.domin = refedges.Get(k).domin;
+	    seg.domout = refedges.Get(k).domout;
+	    seg.tlosurf = refedges.Get(k).tlosurf;
+	    seg.edgenr = refedges.Get(k).edgenr;
+	    seg.surfnr1 = refedges.Get(k).surfnr1;
+	    seg.surfnr2 = refedges.Get(k).surfnr2;
+	    seg.seginfo = 0;
+	    if (k == 1) seg.seginfo = (refedgesinv.Get(k)) ? 2 : 1;
+	    mesh.AddSegment (seg);
+	    //	  (*testout) << "add seg " << seg.p1 << "-" << seg.p2 << endl;
+	  
+	    double maxh = min2 (geometry.GetSurface(seg.surfnr1)->GetMaxH(),
+				geometry.GetSurface(seg.surfnr2)->GetMaxH());
+			      
+	    if (seg.domin != -1)
+	      {
+		const Solid * s1 = 
+		  geometry.GetTopLevelObject(seg.domin) -> GetSolid();
+		maxh = min2 (maxh, s1->GetMaxH());
+		maxh = min2 (maxh, geometry.GetTopLevelObject(seg.domin)->GetMaxH());
+		mesh.RestrictLocalH (p, maxh);
+		mesh.RestrictLocalH (np, maxh);
+	      }
+	    if (seg.domout != -1)
+	      {
+		const Solid * s1 = 
+		  geometry.GetTopLevelObject(seg.domout) -> GetSolid();
+		maxh = min2 (maxh, s1->GetMaxH());
+		maxh = min2 (maxh, geometry.GetTopLevelObject(seg.domout)->GetMaxH());
+		mesh.RestrictLocalH (p, maxh);
+		mesh.RestrictLocalH (np, maxh);
+	      }
+	    if (seg.tlosurf != -1)
+	      {
+		double hi = geometry.GetTopLevelObject(seg.tlosurf) -> GetMaxH();
+		maxh = min2 (maxh, hi);
+		mesh.RestrictLocalH (p, maxh);
+		mesh.RestrictLocalH (np, maxh);
+	      }	  
+	  }
+      
+	p = np;
+	lastpi = thispi;
+      }
+
+#ifdef DEVELOP
+    (*testout) << " eplast = " << lastpi << " = " << p << endl;
+#endif
+  }
+  
+
+
+
+
+
+  void EdgeCalculation :: 
+  StoreShortEdge (const ARRAY<Segment> & refedges,
+		  const ARRAY<int> & refedgesinv,
+		  const ARRAY<Point<3> > & edgepoints,
+		  const ARRAY<double> & curvelength,
+		  int layer,
+		  Mesh & mesh)
+  {
+  
+    // Calculate optimal element-length
+    int i, j, k;
+    PointIndex pi;
+    int ne;
+    Segment seg;
+
+    /*
+      double len, corr, lam;
+      int thispi, lastpi;
+      Point<3> p, np;
+
+
+      const Surface * surf1 = geometry.GetSurface (refedges.Get(1).surfnr1);
+      const Surface * surf2 = geometry.GetSurface (refedges.Get(1).surfnr2);
+
+      len = curvelength.Last();
+      ne = int (len + 0.5);
+      if (ne == 0) ne = 1;
+      if (Dist2 (edgepoints[1], edgepoints.Last()) < 1e-8 && 
+      ne <= 6) 
+      ne = 6;
+      corr = len / ne;
+    */
+
+    // generate initial point
+    Point<3> p = edgepoints[0];
+    PointIndex pi1 = -1;
+    for (pi = PointIndex::BASE; 
+	 pi < mesh.GetNP()+PointIndex::BASE; pi++)
+
+      if (Dist (mesh[pi], p) < 1e-6)
+	{
+	  pi1 = pi;
+	  break;
+	}
+
+    if (pi1 == -1) 
+      {
+	pi1 = mesh.AddPoint (p, layer);
+	meshpoint_tree -> Insert (p, pi1);
+      }
+
+    p = edgepoints.Last();
+    PointIndex pi2 = -1;
+    for (pi = PointIndex::BASE; 
+	 pi < mesh.GetNP()+PointIndex::BASE; pi++)
+
+      if (Dist (mesh[pi], p) < 1e-6)
+	{
+	  pi2 = pi;
+	  break;
+	}
+    if (pi2==-1) 
+      {
+	pi2 = mesh.AddPoint (p, layer);
+	meshpoint_tree -> Insert (p, pi2);
+      }
+
+    /*
+  
+    j = 1;
+    for (i = 1; i <= ne; i++)
+    {
+    while (curvelength[j] < i * corr && j < curvelength.Size()) j++;
+      
+    lam = (i * corr - curvelength[j-1]) / 
+    (curvelength[j] - curvelength[j-1]);
+      
+    np(0) = (1-lam) * edgepoints[j-1](0) + lam * edgepoints[j](0);
+    np(1) = (1-lam) * edgepoints[j-1](1) + lam * edgepoints[j](1);
+    np(2) = (1-lam) * edgepoints[j-1](2) + lam * edgepoints[j](2);
+      
+      
+    thispi = 0;
+    if (i == ne)
+    for (j = 1; j <= mesh.GetNP(); j++)
+    if (Dist(mesh.Point(j), np) < 1e-6)
+    thispi = j;
+      
+    if (!thispi)
+    {
+    ProjectToEdge (surf1, surf2, np);
+    thispi = mesh.AddPoint (np);
+    }
+    */
+  
+    for (k = 1; k <= refedges.Size(); k++)
+      {
+	if (refedgesinv.Get(k))
+	  {
+	    seg.p1 = pi1;
+	    seg.p2 = pi2;
+	  }
+	else
+	  {
+	    seg.p1 = pi2;
+	    seg.p2 = pi1;
+	  }
+
+	seg.si = refedges.Get(k).si;
+	seg.domin = refedges.Get(k).domin;
+	seg.domout = refedges.Get(k).domout;
+	seg.tlosurf = refedges.Get(k).tlosurf;
+	seg.edgenr = refedges.Get(k).edgenr;
+	seg.surfnr1 = refedges.Get(k).surfnr1;
+	seg.surfnr2 = refedges.Get(k).surfnr2;
+	seg.seginfo = 0;
+	if (k == 1) seg.seginfo = (refedgesinv.Get(k)) ? 2 : 1;
+	mesh.AddSegment (seg);
+	//	  (*testout) << "add seg " << seg.p1 << "-" << seg.p2 << endl;
+      }
+  }
+  
+
+
+
+
+
+
+  void EdgeCalculation :: 
+  CopyEdge (const ARRAY<Segment> & refedges,
+	    const ARRAY<int> & refedgesinv,
+	    int copyfromedge, 
+	    const Point<3> & fromstart, const Point<3> & fromend,
+	    const Point<3> & tostart, const Point<3> & toend,
+	    int copyedgeidentification, 
+	    int layer,
+	    Mesh & mesh)
+  {
+    int i, j, k;
+    PointIndex pi;
+
+    // copy start and end points
+    for (i = 1; i <= 2; i++)
+      {
+	Point<3> fromp =
+	  (i == 1) ? fromstart : fromend;
+	Point<3> top =
+	  (i == 1) ? tostart : toend;
+      
+	PointIndex frompi = -1;
+	PointIndex topi = -1;
+	for (pi = PointIndex::BASE; 
+	     pi < mesh.GetNP()+PointIndex::BASE; pi++)
+	  {
+	    if (Dist2 (mesh[pi], fromp) <= 1e-16)
+	      frompi = pi;
+	    if (Dist2 (mesh[pi], top) <= 1e-16)
+	      topi = pi;
+	  }
+
+	if (topi == -1)
+	  {
+	    topi = mesh.AddPoint (top, layer);
+	    meshpoint_tree -> Insert (top, topi);
+	  }
+
+	const Identification & csi = 
+	  (*geometry.identifications.Get(copyedgeidentification));
+
+	if (csi.Identifyable (mesh[frompi], mesh[topi]))
+	  mesh.GetIdentifications().Add(frompi, topi, copyedgeidentification);
+	else if (csi.Identifyable (mesh[topi], mesh[frompi]))
+	  mesh.GetIdentifications().Add(topi, frompi, copyedgeidentification);
+	else
+	  {
+	    cerr << "edgeflw.cpp: should identify, but cannot";
+	    exit(1);
+	  }
+	/*
+	  (*testout) << "Add Identification from CopyEdge, p1 = " 
+	  << mesh[PointIndex(frompi)] << ", p2 = " 
+	  << mesh[PointIndex(topi)] << endl;
+
+	  mesh.GetIdentifications().Add(frompi, topi, copyedgeidentification);
+	*/
+      }
+
+    int oldns = mesh.GetNSeg();
+    for (i = 1; i <= oldns; i++)
+      {
+	// real copy, since array might be reallocated !!
+	const Segment oldseg = mesh.LineSegment(i);
+	if (oldseg.edgenr != copyfromedge)
+	  continue;
+	if (oldseg.seginfo == 0)
+	  continue;
+
+	int pi1 = oldseg.p1;
+	int pi2 = oldseg.p2;
+
+	int npi1 = geometry.identifications.Get(copyedgeidentification)
+	  -> GetIdentifiedPoint (mesh, pi1);
+	int npi2 = geometry.identifications.Get(copyedgeidentification)
+	  -> GetIdentifiedPoint (mesh, pi2);
+
+	Segment seg;
+
+	for (k = 1; k <= refedges.Size(); k++)
+	  {
+	    int inv = refedgesinv.Get(k);
+
+	    // other edge is inverse
+	    if (oldseg.seginfo == 1)
+	      inv = !inv;
+
+	    //	  (*testout) << "inv, now = " << inv << endl;
+
+	    if (inv)
+	      {
+		seg.p1 = npi1;
+		seg.p2 = npi2;
+	      }
+	    else
+	      {
+		seg.p1 = npi2;
+		seg.p2 = npi1;
+	      }
+	    seg.si = refedges.Get(k).si;
+	    seg.domin = refedges.Get(k).domin;
+	    seg.domout = refedges.Get(k).domout;
+	    seg.tlosurf = refedges.Get(k).tlosurf;
+	    seg.edgenr = refedges.Get(k).edgenr;
+	    seg.surfnr1 = refedges.Get(k).surfnr1;
+	    seg.surfnr2 = refedges.Get(k).surfnr2;
+	    seg.seginfo = 0;
+	    if (k == 1) seg.seginfo = refedgesinv.Get(k) ? 2 : 1;
+	    mesh.AddSegment (seg);
+	    //	  (*testout) << "copy seg " << seg.p1 << "-" << seg.p2 << endl;
+#ifdef DEVELOP
+
+	    (*testout) << "copy seg, face = " << seg.si << ": " 
+		       << " inv = " << inv << ", refinv = " << refedgesinv.Get(k)
+		       << mesh.Point(seg.p1) << ", " << mesh.Point(seg.p2) << endl;
+#endif
+
+	  }
+      
+      }   
+  }
+  
+
+
+
+
+
+
+  void EdgeCalculation :: 
+  FindClosedSurfaces (double h, Mesh & mesh)
+  {
+    // if there is no special point at a sphere, one has to add a segment pair
+  
+    int i, j; 
+    int nsol; 
+    int nsurf = geometry.GetNSurf();
+    int layer;
+
+    BitArray pointatsurface (nsurf);
+    Point<3> p1, p2;
+    Vec<3> nv, tv;
+    Solid * tansol;
+    ARRAY<int> tansurfind;
+    //  const Solid * sol;
+
+    nsol = geometry.GetNTopLevelObjects();
+
+
+    pointatsurface.Clear();
+  
+    /*
+      for (i = 1; i <= specpoints.Size(); i++)
+      {
+      int classrep;
+
+      classrep = geometry.GetSurfaceClassRepresentant (specpoints[i].s1);
+      pointatsurface.Set (classrep);
+      classrep = geometry.GetSurfaceClassRepresentant (specpoints[i].s2);
+      pointatsurface.Set (classrep);
+      //      pointatsurface.Set (specpoints[i].s1);
+      //      pointatsurface.Set (specpoints[i].s2);
+      }
+    */
+    for (i = 1; i <= mesh.GetNSeg(); i++)
+      {
+	const Segment & seg = mesh.LineSegment(i);
+	int classrep;
+
+#ifdef DEVELOP      
+	(*testout) << seg.surfnr1 << ", " << seg.surfnr2 << ", si = " << seg.si << endl;
+#endif
+	classrep = geometry.GetSurfaceClassRepresentant (seg.si);
+
+	pointatsurface.Set (classrep);
+      }
+
+  
+    for (i = 0; i < nsurf; i++)
+      {
+	int classrep = geometry.GetSurfaceClassRepresentant (i);
+
+	if (!pointatsurface.Test(classrep))
+	  {
+	    const Surface * s = geometry.GetSurface(i);
+	    p1 = s -> GetSurfacePoint();
+	    s -> GetNormalVector (p1, nv);
+		    
+	    double hloc = 
+	      min2 (s->LocH (p1, 3, 1, h), mesh.GetH(p1));
+
+	    tv = nv.GetNormal ();
+	    tv *=  (hloc / tv.Length());
+	    p2 = p1 + tv;
+	    s->Project (p2);
+	  
+		    
+	    Segment seg1;
+	    seg1.si = i;
+	    seg1.domin = -1;
+	    seg1.domout = -1;
+
+	    Segment seg2;
+	    seg2.si = i;
+	    seg2.domin = -1;
+	    seg2.domout = -1;
+
+	    seg1.surfnr1 = i;
+	    seg2.surfnr1 = i;
+	    seg1.surfnr2 = i;
+	    seg2.surfnr2 = i;
+
+	    for (j = 0; j < nsol; j++)
+	      {
+		if (geometry.GetTopLevelObject(j)->GetSurface())
+		  continue;
+
+		const Solid * sol = geometry.GetTopLevelObject(j)->GetSolid();
+		sol -> TangentialSolid (p1, tansol);
+		layer = geometry.GetTopLevelObject(j)->GetLayer();
+
+		if (tansol)
+		  {
+		    tansol -> GetSurfaceIndices (tansurfind);
+		
+		    if (tansurfind.Size() == 1 && tansurfind.Get(1) == i)
+		      {
+			if (!tansol->VectorIn(p1, nv))
+			  {
+			    seg1.domin = j;
+			    seg2.domin = j;
+			    seg1.tlosurf = j;
+			    seg2.tlosurf = j;
+			  }
+			else
+			  {
+			    seg1.domout = j;
+			    seg2.domout = j;
+			    seg1.tlosurf = j;
+			    seg2.tlosurf = j;
+			  }
+			//        seg.s2 = i;
+			//        seg.invs1 = surfaces[i] -> Inverse();
+			//        seg.invs2 = ! (surfaces[i] -> Inverse());
+		      }
+		    delete tansol;
+		  }
+	      }
+
+
+	    if (seg1.domin != -1 || seg1.domout != -1)
+	      {
+		mesh.AddPoint (p1, layer);
+		mesh.AddPoint (p2, layer);
+		seg1.p1 = mesh.GetNP()-1;
+		seg1.p2 = mesh.GetNP();
+		seg2.p2 = mesh.GetNP()-1;
+		seg2.p1 = mesh.GetNP();
+		seg1.geominfo[0].trignum = 1;
+		seg1.geominfo[1].trignum = 1;
+		seg2.geominfo[0].trignum = 1;
+		seg2.geominfo[1].trignum = 1;
+		mesh.AddSegment (seg1);
+		mesh.AddSegment (seg2);
+
+		PrintMessage (5, "Add line segment to smooth surface");
+
+#ifdef DEVELOP
+		(*testout) << "Add segment at smooth surface " << i;
+		if (i != classrep) (*testout) << ", classrep = " << classrep;
+		(*testout) << ": "
+			   << mesh.Point (mesh.GetNP()-1) << " - "
+			   << mesh.Point (mesh.GetNP()) << endl;
+#endif
+	      }
+	  }
+      }
+  }
+
+}
diff --git a/Netgen/libsrc/csg/edgeflw_old.cpp b/Netgen/libsrc/csg/edgeflw_old.cpp
new file mode 100644
index 0000000000..5321dfd17d
--- /dev/null
+++ b/Netgen/libsrc/csg/edgeflw_old.cpp
@@ -0,0 +1,1405 @@
+#include <mystdlib.h>
+#include <meshing.hpp>
+#include <csg.hpp>
+
+#undef DEVELOP
+
+namespace netgen
+{
+
+  EdgeCalculation :: 
+  EdgeCalculation (const CSGeometry & ageometry,
+		   ARRAY<SpecialPoint> & aspecpoints)
+    : geometry(ageometry), specpoints(aspecpoints)
+  {
+    ;
+  }
+
+  EdgeCalculation :: ~EdgeCalculation ()
+  { ; }
+
+  void EdgeCalculation :: Calc(double h, Mesh & mesh)
+  {
+    PrintMessage (1, "Find edges");
+    PushStatus ("Find edges");
+
+    CalcEdges1 (h, mesh);
+    SplitEqualOneSegEdges (mesh);
+    FindClosedSurfaces (h, mesh);
+    PrintMessage (3, cntedge, " edges found");
+
+    PopStatus ();
+  }
+
+
+
+
+  void EdgeCalculation :: CalcEdges1 (double h, Mesh & mesh)
+  {
+    ARRAY<SpecialPoint> hsp(specpoints.Size());
+    ARRAY<SpecialPoint> startpoints, endpoints;
+
+    int i, j, k, l, hi, pos, ep, ne;
+    int layer;
+
+    Vec<3> a1, a2, t, n, m;
+    Point<3> p, np, pnp, hp;
+
+    Segment seg;
+    int pi1, s1, s2;
+    int lastpi, thispi;
+
+    ARRAY<Point<3> > edgepoints;
+    ARRAY<double> curvelength;
+    int copyedge, copyfromedge, copyedgeidentification;
+
+    ARRAY<int> locsurfind;
+
+    double len, corr, lam;
+    double steplen, cursteplen, loch, hd;
+
+    int checkedcopy = 0;
+
+    double size = geometry.MaxSize(); // globflags.GetNumFlag ("maxsize", 500);
+    double epspointdist2 = size * 1e-6; // globflags.GetNumFlag ("epspointdist", size * 1e-6);
+    epspointdist2 = sqr (epspointdist2);
+
+
+    Solid * locsol;
+
+
+    // copy special points to work with
+    for (i = 0; i < specpoints.Size(); i++)
+      hsp[i] = specpoints[i];
+
+
+    cntedge = 0;
+
+    while (hsp.Size())
+      {
+	SetThreadPercent(100 - 100 * double (hsp.Size()) / specpoints.Size());
+
+	edgepoints.SetSize (0);
+	curvelength.SetSize (0);
+      
+
+	pi1 = 0;
+	copyedge = 0;
+	// identifyable point available ?
+
+	//      (*testout) << endl;
+
+	for (i = 1; i <= geometry.identifications.Size() && !pi1; i++)
+	  {
+	    for (j = checkedcopy+1; j <= startpoints.Size() && !pi1; j++)
+	      {
+
+		if (geometry.identifications.Get(i)->IdentifyableCandidate (startpoints.Get(j)))
+		
+		  {
+		    int pi1cand = 0;
+		    double mindist = 1e10;
+		  
+		    for (k = 1; k <= hsp.Size() && !pi1; k++)
+		      {
+#ifdef DEVELOP
+			(*testout) << "check kand = " << hsp.Get(k).p 
+				   << ", v = " << hsp.Get(k).v 
+				   << endl;		      
+#endif
+			if (geometry.identifications.Get(i)
+			    ->Identifyable(startpoints.Get(j), hsp.Get(k)) ||
+			    geometry.identifications.Get(i)
+			    ->Identifyable(hsp.Get(k), startpoints.Get(j)))
+			  {
+
+#ifdef DEVELOP
+			    (*testout) << "identifiable, dist = "
+				       << Dist (startpoints.Get(j).p, hsp.Get(k).p) << endl;
+#endif
+
+			    if (Dist (startpoints.Get(j).p, hsp.Get(k).p) < mindist)
+			      {
+				mindist = Dist (startpoints.Get(j).p, hsp.Get(k).p);
+				pi1cand = k;
+			      }
+			    /*
+			      pi1 = k;
+			      copyedge = 1;
+			      copyfromedge = j;
+			      copyedgeidentification = i;
+			  
+			      (*testout) << "copy edge startpoint from "
+			      << startpoints.Get(j).p << " - " 
+			      << startpoints.Get(j).v 
+			      << " to " 
+			      << hsp.Get(k).p << " - " << hsp.Get(k).v << endl;
+			    */
+			  }
+		      }
+
+		    if (pi1cand)
+		      {
+			pi1 = pi1cand;
+			copyedge = 1;
+			copyfromedge = j;
+			copyedgeidentification = i;
+#ifdef DEVELOP
+			(*testout) << "copy edge startpoint from "
+				   << startpoints.Get(j).p << " - " 
+				   << startpoints.Get(j).v 
+				   << " to " 
+				   << hsp.Get(pi1).p << " - " << hsp.Get(pi1).v << endl;
+#endif
+		      }
+		  }
+	      }
+	  }
+      
+      
+	// cannot copy from other ege ?
+	if (!pi1)
+	  checkedcopy = startpoints.Size();
+      
+	// unconditional special point available ?
+	if (!pi1)
+	  for (i = 1; i <= hsp.Size() && pi1 == 0; i++)
+	    if (hsp.Get(i).unconditional == 1)
+	      pi1 = i;
+ 
+     
+	if (!pi1)
+	  {
+	    // only unconditional points available, choose first
+	    pi1 = 1;
+	  }
+
+	layer = hsp.Get(pi1).GetLayer();
+      
+
+	if (!hsp.Get(pi1).unconditional)
+	  {
+	    hsp.Elem(pi1).unconditional = 1;
+	    for (i = 1; i <= hsp.Size(); i++)
+	      if (i != pi1 && Dist (hsp.Get(pi1).p, hsp.Get(i).p) < 1e-8 &&
+		  (hsp.Get(pi1).v + hsp.Get(i).v).Length() < 1e-4)
+		{
+		  // opposite direction
+		  hsp.Elem(i).unconditional = 1;
+		}
+	  }
+
+	cntedge++;
+	startpoints.Append (hsp.Get(pi1));
+
+#ifdef DEVELOP
+	(*testout) << "edge nr " << cntedge << endl;
+	(*testout) << "start followedge: p1 = " << hsp.Get(pi1).p << ", v = " << hsp.Get(pi1).v << endl;
+#endif
+
+	FollowEdge (pi1, ep, pos, hsp, h, mesh,
+		    edgepoints, curvelength);
+
+
+	if (multithread.terminate)
+	  return;
+      
+	if (!ep)
+	  {
+	    // ignore starting point
+	    hsp.DeleteElement (pi1);
+	    continue;
+	  }
+
+
+
+	endpoints.Append (hsp.Get(ep));
+
+
+	double elen = 0;
+	for (i = 1; i <= edgepoints.Size()-1; i++)
+	  elen += Dist (edgepoints.Get(i), edgepoints.Get(i+1));
+
+
+	int shortedge = 0;
+	for (i = 1; i <= geometry.identifications.Size(); i++)
+	  if (geometry.identifications.Get(i)->ShortEdge(hsp.Get(pi1), hsp.Get(ep)))
+	    shortedge = 1;
+	(*testout) << "shortedge = " << shortedge << endl;
+
+
+	if (!shortedge)
+	  {
+	    mesh.RestrictLocalHLine (Point3d (hsp.Get(pi1).p), 
+				     Point3d (hsp.Get(ep).p), 
+				     elen / mparam.segmentsperedge);
+	  }
+      
+	s1 = hsp.Get(pi1).s1;
+	s2 = hsp.Get(pi1).s2;
+
+
+	// delete initial, terminal and conditional points
+
+#ifdef DEVELOP
+	(*testout) << "terminal point: p = " << hsp.Get(ep).p << ", v = " << hsp.Get(ep).v << endl;      
+#endif
+	if (ep > pi1)
+	  {
+	    hsp.DeleteElement (ep);
+	    hsp.DeleteElement (pi1);
+	  }
+	else
+	  {
+	    hsp.DeleteElement (pi1);
+	    hsp.DeleteElement (ep);
+	  }
+
+
+	for (j = 1; j <= edgepoints.Size()-1; j++)
+	  {
+	    p = edgepoints.Get(j);
+	    np = Center (p, edgepoints.Get(j+1));
+	    hd = Dist2 (p, np);
+ 
+	    for (i = 1; i <= hsp.Size(); i++)
+	      if ( hsp.Get(i).HasSurfaces (s1, s2) &&
+		   hsp.Get(i).unconditional == 0 &&
+		   Dist2 (np, hsp.Get(i).p) < 1.2 * hd)
+		{
+		  hsp.DeleteElement (i);
+		  i--;
+		}
+	  }
+
+      
+	ARRAY<Segment> refedges;
+	ARRAY<int> refedgesinv;
+      
+
+	AnalyzeEdge (s1, s2, pos, layer,
+		     edgepoints,
+		     refedges, refedgesinv);
+
+	for (i = 1; i <= refedges.Size(); i++)
+	  refedges.Elem(i).edgenr = cntedge;
+
+
+#ifdef DEVELOP
+	(*testout) << "edge " << cntedge << endl
+		   << "startp: " << startpoints.Last().p 
+		   << ", v = " << startpoints.Last().v << endl
+		   << "copy = " << copyedge << endl
+		   << refedges.Size() << " refedges: ";
+	for (i = 1; i <= refedges.Size(); i++)
+	  (*testout) << " " << refedges.Get(i).si;
+	(*testout) << endl;
+	(*testout) << "inv[1] = " << refedgesinv.Get(1) << endl;
+#endif
+      
+	if (!copyedge)
+	  {
+	    int oldnseg = mesh.GetNSeg();
+
+	    if (!shortedge)
+	      StoreEdge (refedges, refedgesinv, 
+			 edgepoints, curvelength, layer, mesh);
+	    else
+	      StoreShortEdge (refedges, refedgesinv, 
+			      edgepoints, curvelength, layer, mesh);
+
+
+	    /*
+	      for (i = oldnseg+1; i <= mesh.GetNSeg(); i++)
+	      for (j = 1; j <= oldnseg; j++)
+	      {
+	      const Point<3> & l1p1 = mesh.Point (mesh.LineSegment(i).p1);
+	      const Point<3> & l1p2 = mesh.Point (mesh.LineSegment(i).p2);
+	      const Point<3> & l2p1 = mesh.Point (mesh.LineSegment(j).p1);
+	      const Point<3> & l2p2 = mesh.Point (mesh.LineSegment(j).p2);
+	      Vec<3> vl1(l1p1, l1p2);
+	      for (double lamk = 0; lamk <= 1; lamk += 0.1)
+	      {
+	      Point<3> l2p = l1p1 + lamk * vl1;
+	      double dist = sqrt (MinDistLP2 (l2p1, l2p2, l2p));
+	      if (dist > 1e-12)
+	      mesh.RestrictLocalH (l2p, 3*dist);
+	      }
+	      }
+	    */
+	  }
+	else
+	  {
+	    CopyEdge (refedges, refedgesinv,
+		      copyfromedge, 
+		      startpoints.Get(copyfromedge).p,
+		      endpoints.Get(copyfromedge).p,
+		      edgepoints.Get(1), edgepoints.Last(),
+		      copyedgeidentification, 
+		      layer,
+		      mesh);
+	  }
+
+      }
+  }
+
+
+
+  /*
+    If two or more edges share the same initial and end-points,
+    then they need at least two segments 
+  */
+  void EdgeCalculation ::
+  SplitEqualOneSegEdges (Mesh & mesh)
+  {
+    int i, j;
+    SegmentIndex si;
+    PointIndex pi;
+
+    ARRAY<int> osedges(cntedge);
+    INDEX_2_HASHTABLE<int> osedgesht (cntedge+1);
+
+    osedges = 2;
+
+    // count segments on edges
+    for (si = 0; si < mesh.GetNSeg(); si++)
+      {
+	const Segment & seg = mesh[si];
+	if (seg.seginfo && seg.edgenr >= 1 && seg.edgenr <= cntedge)
+	  osedges.Elem(seg.edgenr)--;
+      }
+
+    (*testout) << "osedges  = " << osedges << endl;
+
+    // flag one segment edges
+    for (i = 0; i < cntedge; i++)
+      osedges[i] = (osedges[i] > 0) ? 1 : 0;
+
+    (*testout) << "osedges, now  = " << osedges << endl;
+
+    for (si = 0; si < mesh.GetNSeg(); si++)
+      {
+	const Segment & seg = mesh[si];
+	if (seg.seginfo && seg.edgenr >= 1 && seg.edgenr <= cntedge)
+	  {
+	    if (osedges.Get(seg.edgenr))
+	      {
+		INDEX_2 i2(seg.p1, seg.p2);
+		i2.Sort ();
+		if (osedgesht.Used (i2))
+		  osedgesht.Set (i2, 2);
+		else
+		  osedgesht.Set (i2, 1);
+	      }
+	  }
+      }
+
+
+    // one edge 1 segment, other 2 segments 
+    // yes, it happens !
+  
+    for (i = 1; i <= osedgesht.GetNBags(); i++)
+      for (j = 1; j <= osedgesht.GetBagSize(i); j++)
+	{
+	  INDEX_2 i2; 
+	  int val;
+	  osedgesht.GetData (i, j, i2, val);
+
+	  const Point<3> & p1 = mesh[PointIndex(i2.I1())];
+	  const Point<3> & p2 = mesh[PointIndex(i2.I2())];
+	  Vec<3> v = p2 - p1;
+	  double vlen = v.Length();
+	  v /= vlen;
+	  for (pi = PointIndex::BASE; 
+	       pi < mesh.GetNP()+PointIndex::BASE; pi++)
+
+	    if (pi != i2.I1() && pi != i2.I2())
+	      {
+		const Point<3> & p = mesh[pi];
+		Vec<3> v2 = p - p1;
+		double lam = (v2 * v);
+		if (lam > 0 && lam < vlen)
+		  {
+		    Point<3> hp = p1 + lam * v;
+		    if (Dist (p, hp) < 1e-4 * vlen)
+		      {
+			PrintSysError ("Point on edge !!!");
+			cout << "seg: " << i2 << ", p = " << pi << endl;
+			osedgesht.Set (i2, 2);		      
+		      }
+		  }
+	      }
+	}
+
+
+    // insert new points
+    osedges = -1;
+
+    int nseg = mesh.GetNSeg();
+    for (si = 0; si < nseg; si++)
+      {
+	const Segment & seg = mesh[si];
+	if (seg.seginfo && seg.edgenr >= 1 && seg.edgenr <= cntedge)
+	  {
+	    INDEX_2 i2(seg.p1, seg.p2);
+	    i2.Sort ();
+	    if (osedgesht.Used (i2) &&
+		osedgesht.Get (i2) == 2 &&
+		osedges.Elem(seg.edgenr) == -1)
+	      {
+		Point<3> newp = Center (mesh[PointIndex(seg.p1)],
+					mesh[PointIndex(seg.p2)]);
+
+		ProjectToEdge (geometry.GetSurface(seg.surfnr1), 
+			       geometry.GetSurface(seg.surfnr2), 
+			       newp);
+
+		osedges.Elem(seg.edgenr) = 
+		  mesh.AddPoint (newp, mesh[PointIndex(seg.p1)].GetLayer());
+	      }
+	  }
+      }
+
+
+    for (i = 1; i <= nseg; i++)
+      {
+	Segment & seg = mesh.LineSegment (i);
+	if (seg.edgenr >= 1 && seg.edgenr <= cntedge)
+	  {
+	    if (osedges.Get(seg.edgenr) != -1)
+	      {
+		Segment newseg = seg;
+		newseg.p1 = osedges.Get(seg.edgenr);
+		seg.p2 = osedges.Get(seg.edgenr);
+		mesh.AddSegment (newseg);
+	      }
+	  }
+      }
+
+  }
+
+
+
+  void EdgeCalculation :: 
+  FollowEdge (int pi1, int & ep, int & pos,
+	      const ARRAY<SpecialPoint> & hsp,
+	      double h, const Mesh & mesh,
+	      ARRAY<Point<3> > & edgepoints,
+	      ARRAY<double> & curvelength)
+  {
+    int i, j, s1, s2;
+    double len, steplen, cursteplen, loch;
+    Point<3> p, np, pnp;
+    Vec<3> a1, a2, t;
+
+
+    double size = geometry.MaxSize();  
+    double epspointdist2 = size * 1e-6;
+    epspointdist2 = sqr (epspointdist2);
+    int uselocalh = mparam.uselocalh;
+
+
+    s1 = hsp.Get(pi1).s1;
+    s2 = hsp.Get(pi1).s2;
+  
+    p = hsp.Get(pi1).p;
+    geometry.GetSurface(s1) -> CalcGradient (p, a1);
+    geometry.GetSurface(s2) -> CalcGradient (p, a2);
+
+    t = Cross (a1, a2);
+    t.Normalize();
+
+    pos = (hsp.Get(pi1).v * t) > 0;
+    if (!pos) t *= -1;
+
+  
+    edgepoints.Append (p);
+    curvelength.Append (0);
+    len = 0;
+
+    loch = min2 (geometry.GetSurface(s1) -> LocH (p, 3, 1, h), 
+		 geometry.GetSurface(s2) -> LocH (p, 3, 1, h));
+  
+  
+  
+    if (uselocalh)
+      {
+	double lh = mesh.GetH(p);
+	if (lh < loch)
+	  loch = lh;
+      }
+
+    steplen = 0.1 * loch;
+  
+    do
+      {
+	if (multithread.terminate)
+	  return;
+      
+	if (fabs (p(0)) + fabs (p(1)) + fabs (p(2)) > 10000)
+	  {
+	    ep = 0;
+	    PrintWarning ("Give up line");
+	    break;
+	  }
+
+	if (steplen > 0.1 * loch) steplen = 0.1 * loch;
+      
+	steplen *= 2;
+	do
+	  {
+	    steplen *= 0.5;
+	    np = p + steplen * t;
+	    pnp = np;
+	    ProjectToEdge (geometry.GetSurface(s1), 
+			   geometry.GetSurface(s2), pnp);
+	  }
+	while (Dist (np, pnp) > 0.1 * steplen);
+      
+	cursteplen = steplen;
+	if (Dist (np, pnp) < 0.01 * steplen) steplen *= 2;
+      
+ 
+	np = pnp;
+      
+#ifdef MYGRAPH
+	if (silentflag <= 2)
+	  {
+	    MyLine3D (p, np, rot);
+	    MyDraw ();
+	  }
+#endif      
+
+	ep = 0;
+      
+	double hvtmin = 1.5 * cursteplen;
+      
+	Box<3> boxp (p - (2 * cursteplen) * Vec<3> (1, 1, 1),
+		     p + (2 * cursteplen) * Vec<3> (1, 1, 1));
+
+	for (i = 1; i <= hsp.Size(); i++)
+	  // if ( i != pi1 && hsp.Get(i).HasSurfaces (s1, s2) )
+	  {
+	    if (!boxp.IsIn (hsp.Get(i).p))
+	      continue;
+	  
+	    Vec<3> hv = hsp.Get(i).p - p;
+	    if (hv.Length2() > 9 * cursteplen * cursteplen)
+	      continue;
+
+	    /*
+	    if (!hsp.Get(i).HasSurfaces (s1, s2))
+	      continue;                  // test for dalibor-problem
+	    */
+
+	    double hvt = hv * t;
+	    hv -= hvt * t;
+	  
+	    if (hv.Length() < 0.2 * cursteplen &&
+		hvt > 0 && 
+		//		  hvt < 1.5 * cursteplen &&
+		hvt < hvtmin && 
+		hsp.Get(i).unconditional == 1 &&
+		(hsp.Get(i).v + t).Length() < 0.4  ) 
+	      {
+		Point<3> hep = hsp.Get(i).p;
+		ProjectToEdge (geometry.GetSurface(s1), 
+			       geometry.GetSurface(s2), hep);            
+	      
+	      
+		if (Dist2 (hep, hsp.Get(i).p) < epspointdist2 )
+		  {
+		    geometry.GetSurface(s1) -> CalcGradient (hep, a1);
+		    geometry.GetSurface(s2) -> CalcGradient (hep, a2);
+		    Vec<3> ept = Cross (a1, a2);
+		    ept /= ept.Length();
+		    if (!pos) ept *= -1;
+		  
+		    if ( (hsp.Get(i).v + ept).Length() < 1e-4 )
+		      {
+			np = hsp.Get(i).p;
+			ep = i;
+			hvtmin = hvt;
+			//			  break;
+		      }
+		  }
+	      }
+	  }
+
+	loch = min2 (geometry.GetSurface(s1) -> LocH (np, 3, 1, h), 
+		     geometry.GetSurface(s2) -> LocH (np, 3, 1, h));
+
+	if (uselocalh)
+	  {
+	    double lh = mesh.GetH(np);
+	    if (lh < loch)
+	      loch = lh;
+	  }
+      
+      
+	len += Dist (p, np) / loch;
+	edgepoints.Append (np);
+	curvelength.Append (len);
+      
+	p = np;
+      
+	geometry.GetSurface(s1) -> CalcGradient (p, a1);
+	geometry.GetSurface(s2) -> CalcGradient (p, a2);
+	t = Cross (a1, a2);
+	t.Normalize();
+	if (!pos) t *= -1;
+      }
+    while (! ep);
+  }
+
+
+
+
+
+
+
+  void EdgeCalculation :: 
+  AnalyzeEdge (int s1, int s2, int pos, int layer,
+	       const ARRAY<Point<3> > & edgepoints,
+	       ARRAY<Segment> & refedges,
+	       ARRAY<int> & refedgesinv)
+  {
+    int i, j, k, l;
+    int hi;
+    Point<3> hp;
+    Vec<3> t, a1, a2, m, n;
+    Segment seg;
+    Solid * locsol;
+    ARRAY<int> locsurfind;
+
+    /*
+      int pi1 = 0, pi2 = 0;
+      extern Mesh * mesh;
+      for (i = 1; i <= mesh->GetNP(); i++)
+      {
+      if (Dist2 (edgepoints.Get(1), mesh->Point(i)) < 1e-12)
+      pi1 = i;
+      if (Dist2 (edgepoints.Last(), mesh->Point(i)) < 1e-12)
+      pi2 = i;
+      }
+      (*testout) << "Analyze edge: " << pi1 << " - " << pi2 << ", pts = " << edgepoints.Size() << endl;
+      (*testout) << "p1 = " << edgepoints.Get(1) << " pl = " << edgepoints.Last() << endl;
+    */
+    int debug = 0;
+    /*
+      Dist2 (Point<3> (2.69642, 1.1866, 2.03), edgepoints.Get(1)) < 1e-6 ||
+      Dist2 (Point<3> (2.69642, 1.1866, 2.03), edgepoints.Last()) < 1e-6;
+    */
+
+    if (debug)
+      {
+	//      (*testout) << "tubious edge !!!" << endl;
+	(*testout) << "s1, s2 = " << s1 << " - " << s2 << endl;
+      }
+
+    refedges.SetSize(0);
+    refedgesinv.SetSize(0);
+    hp = Center (edgepoints.Get(1), edgepoints.Get(2));
+    ProjectToEdge (geometry.GetSurface(s1), geometry.GetSurface(s2), hp);
+
+    geometry.GetSurface(s1) -> CalcGradient (hp, a1);
+    geometry.GetSurface(s2) -> CalcGradient (hp, a2);
+    t = Cross (a1, a2);
+    t.Normalize();
+    if (!pos) t *= -1;    
+  
+    (*testout) << "t = " << t << endl;
+
+    for (i = 0; i < geometry.GetNTopLevelObjects(); i++)
+      {
+	(*testout) << "layer = " << layer 
+		   << ", tlo-layer = " << geometry.GetTopLevelObject(i)->GetLayer() << endl;
+	if (geometry.GetTopLevelObject(i)->GetLayer() != layer) 
+	  continue;
+      
+	const Solid * sol = geometry.GetTopLevelObject(i)->GetSolid();
+	const Surface * surf = geometry.GetTopLevelObject(i)->GetSurface();
+
+	sol -> TangentialSolid (hp, locsol);
+	if (!locsol) continue;
+
+	BoxSphere<3> boxp (hp, hp);
+	boxp.Increase (1e-5);
+	boxp.CalcDiamCenter();
+      
+	ReducePrimitiveIterator rpi(boxp);
+	UnReducePrimitiveIterator urpi;
+      
+	((Solid*)locsol) -> IterateSolid (rpi);
+
+	locsol -> CalcSurfaceInverse ();
+      
+
+	if (!surf)
+	  {
+	    locsol -> GetSurfaceIndices (locsurfind);
+	  }
+	else
+	  {
+	    /*
+	      if (fabs (surf->CalcFunctionValue (hp)) < 1e-6)
+	      continue;
+	    */
+	    locsurfind.SetSize(1);
+	    locsurfind[0] = -1;
+	    for (j = 0; j < geometry.GetNSurf(); j++)
+	      if (geometry.GetSurface(j) == surf)
+		{
+		  locsurfind[0] = j;
+		  //		      geometry.GetSurfaceClassRepresentant(j);
+		  break;
+		}
+	  }
+
+	((Solid*)locsol) -> IterateSolid (urpi);
+
+      
+	if (debug)
+	  (*testout) << "edge of tlo " << i << ", has " << locsurfind.Size() << " faces." << endl;
+      
+
+	for (j = locsurfind.Size()-1; j >= 0; j--)
+	  if (fabs (geometry.GetSurface(locsurfind[j])
+		    ->CalcFunctionValue (hp) ) > 1e-6)
+	    locsurfind.DeleteElement(j+1);
+      
+	if (debug)
+	  (*testout) << locsurfind.Size() << " faces on hp" << endl;
+
+	for (j = 0; j < locsurfind.Size(); j++)
+	  {      
+	    int lsi = locsurfind[j];
+	    int rlsi = geometry.GetSurfaceClassRepresentant(lsi);
+	  
+	    Vec<3> rn;
+
+	    // n is outer normal to solid
+	    geometry.GetSurface(lsi) -> GetNormalVector (hp, n);
+	    if (geometry.GetSurface (lsi)->Inverse())
+	      n *= -1;
+	  
+	    if (fabs (t * n) > 1e-4) continue;
+	    if (debug)
+	      {
+		(*testout) << "face " << locsurfind.Get(j) << ", rep = " << rlsi 
+			   << " has (t*n) = " << (t*n) << endl;
+		(*testout) << "n = " << n << endl;
+	      }
+	  
+	    // rn is normal to class representant
+	    geometry.GetSurface(rlsi) -> GetNormalVector (hp, rn);
+	  
+	    int sameasref = ((n * rn) > 0);
+	  
+	    m = Cross (t, rn);
+	    m.Normalize();
+	  
+
+	    for (k = 1; k <= 2; k ++)
+	      {
+		bool edgeinv = (k == 2);
+	      
+		if (debug)
+		  {
+		    (*testout) << "onface(" << hp << ", " << m << ")= " 
+			       << locsol->OnFace (hp, m);
+		    (*testout) << " vec2in = "
+			       << locsol -> VectorIn2 (hp, m, n) << " and " 
+			       << locsol -> VectorIn2 (hp, m, -1 * n) << endl;
+		  }
+
+		//	      if (locsol -> OnFace (hp, m))
+		if (locsol -> VectorIn2 (hp, m, n) == 0 &&
+		    locsol -> VectorIn2 (hp, m, -1 * n) == 1)
+		  {
+		    hi = 0;
+		    for (l = 1; l <= refedges.Size(); l++)
+		      {
+			if (refedges.Get(l).si == rlsi &&
+			    refedgesinv.Get(l) == edgeinv)
+			  hi = l;
+		      }
+		  
+		    if (!hi)
+		      {
+			seg.si = rlsi;
+			seg.domin = -1;
+			seg.domout = -1;
+			seg.tlosurf = -1;
+			seg.surfnr1 = s1;
+			seg.surfnr2 = s2;
+			hi = refedges.Append (seg);
+			refedgesinv.Append (edgeinv);
+		      }
+		  
+		    if (!surf)
+		      {
+			if (sameasref)
+			  refedges.Elem(hi).domin = i;
+			else 
+			  refedges.Elem(hi).domout = i;
+		      }
+		    else
+		      refedges.Elem(hi).tlosurf = i;
+
+		    if (debug)
+		      (*testout) << "add ref seg:" 
+				 << "si = " << refedges.Get(hi).si
+				 << ", domin = " << refedges.Get(hi).domin
+				 << ", domout = " << refedges.Get(hi).domout
+				 << ", surfnr1/2 = " << refedges.Get(hi).surfnr1
+				 << ", " << refedges.Get(hi).surfnr2
+				 << ", inv = " << refedgesinv.Get(hi) 
+				 << ", refedgenr = " << hi
+				 << endl;
+		  }
+		m *= -1;
+	      } 
+	  }
+	delete locsol;          
+      }
+  }
+
+
+
+  void EdgeCalculation :: 
+  StoreEdge (const ARRAY<Segment> & refedges,
+	     const ARRAY<int> & refedgesinv,
+	     const ARRAY<Point<3> > & edgepoints,
+	     const ARRAY<double> & curvelength,
+	     int layer,
+	     Mesh & mesh)
+  {
+  
+    // Calculate optimal element-length
+    int i, j, k;
+    PointIndex pi;
+    int ne;
+
+    double len, corr, lam;
+    PointIndex thispi, lastpi;
+    Point<3> p, np;
+    Segment seg;
+
+
+    const Surface * surf1 = geometry.GetSurface (refedges.Get(1).surfnr1);
+    const Surface * surf2 = geometry.GetSurface (refedges.Get(1).surfnr2);
+
+    len = curvelength.Last();
+    ne = int (len + 0.5);
+    if (ne == 0) ne = 1;
+    if (Dist2 (edgepoints.Get(1), edgepoints.Last()) < 1e-8 && 
+	ne <= 6) 
+      ne = 6;
+    corr = len / ne;
+
+    // generate initial point
+    p = edgepoints.Get(1);
+    lastpi = -1;
+    for (pi = PointIndex::BASE; 
+	 pi < mesh.GetNP()+PointIndex::BASE; pi++)
+      if (Dist (mesh[pi], p) < 1e-6)
+	{
+	  lastpi = pi;
+	  break;
+	}
+
+    if (lastpi == -1)
+      lastpi = mesh.AddPoint (p, layer);
+
+  
+    j = 1;
+    for (i = 1; i <= ne; i++)
+      {
+	while (curvelength.Get(j) < i * corr && j < curvelength.Size()) j++;
+      
+	lam = (i * corr - curvelength.Get(j-1)) / 
+	  (curvelength.Get(j) - curvelength.Get(j-1));
+      
+	np(0) = (1-lam) * edgepoints.Get(j-1)(0) + lam * edgepoints.Get(j)(0);
+	np(1) = (1-lam) * edgepoints.Get(j-1)(1) + lam * edgepoints.Get(j)(1);
+	np(2) = (1-lam) * edgepoints.Get(j-1)(2) + lam * edgepoints.Get(j)(2);
+      
+      
+	thispi = -1;
+	if (i == ne)
+	  for (pi = PointIndex::BASE; 
+	       pi < mesh.GetNP()+PointIndex::BASE; pi++)
+	    if (Dist(mesh[pi], np) < 1e-6)
+	      thispi = pi;
+      
+	if (thispi == -1)
+	  {
+	    ProjectToEdge (surf1, surf2, np);
+	    thispi = mesh.AddPoint (np, layer);
+	  }
+
+	for (k = 1; k <= refedges.Size(); k++)
+	  {
+	    if (refedgesinv.Get(k))
+	      {
+		seg.p1 = lastpi;
+		seg.p2 = thispi;
+	      }
+	    else
+	      {
+		seg.p1 = thispi;
+		seg.p2 = lastpi;
+	      }
+	    seg.si = refedges.Get(k).si;
+	    seg.domin = refedges.Get(k).domin;
+	    seg.domout = refedges.Get(k).domout;
+	    seg.tlosurf = refedges.Get(k).tlosurf;
+	    seg.edgenr = refedges.Get(k).edgenr;
+	    seg.surfnr1 = refedges.Get(k).surfnr1;
+	    seg.surfnr2 = refedges.Get(k).surfnr2;
+	    seg.seginfo = 0;
+	    if (k == 1) seg.seginfo = (refedgesinv.Get(k)) ? 2 : 1;
+	    mesh.AddSegment (seg);
+	    //	  (*testout) << "add seg " << seg.p1 << "-" << seg.p2 << endl;
+	  
+	    double maxh = min2 (geometry.GetSurface(seg.surfnr1)->GetMaxH(),
+				geometry.GetSurface(seg.surfnr2)->GetMaxH());
+			      
+	    if (seg.domin != -1)
+	      {
+		const Solid * s1 = 
+		  geometry.GetTopLevelObject(seg.domin) -> GetSolid();
+		maxh = min2 (maxh, s1->GetMaxH());
+		maxh = min2 (maxh, geometry.GetTopLevelObject(seg.domin)->GetMaxH());
+		mesh.RestrictLocalH (p, maxh);
+		mesh.RestrictLocalH (np, maxh);
+	      }
+	    if (seg.domout != -1)
+	      {
+		const Solid * s1 = 
+		  geometry.GetTopLevelObject(seg.domout) -> GetSolid();
+		maxh = min2 (maxh, s1->GetMaxH());
+		maxh = min2 (maxh, geometry.GetTopLevelObject(seg.domout)->GetMaxH());
+		mesh.RestrictLocalH (p, maxh);
+		mesh.RestrictLocalH (np, maxh);
+	      }
+	    if (seg.tlosurf != -1)
+	      {
+		double hi = geometry.GetTopLevelObject(seg.tlosurf) -> GetMaxH();
+		maxh = min2 (maxh, hi);
+		mesh.RestrictLocalH (p, maxh);
+		mesh.RestrictLocalH (np, maxh);
+	      }	  
+	  }
+      
+	p = np;
+	lastpi = thispi;
+      }
+
+#ifdef DEVELOP
+    (*testout) << " eplast = " << lastpi << " = " << p << endl;
+#endif
+  }
+  
+
+
+
+
+
+  void EdgeCalculation :: 
+  StoreShortEdge (const ARRAY<Segment> & refedges,
+		  const ARRAY<int> & refedgesinv,
+		  const ARRAY<Point<3> > & edgepoints,
+		  const ARRAY<double> & curvelength,
+		  int layer,
+		  Mesh & mesh)
+  {
+  
+    // Calculate optimal element-length
+    int i, j, k;
+    PointIndex pi;
+    int ne;
+    Segment seg;
+
+    /*
+      double len, corr, lam;
+      int thispi, lastpi;
+      Point<3> p, np;
+
+
+      const Surface * surf1 = geometry.GetSurface (refedges.Get(1).surfnr1);
+      const Surface * surf2 = geometry.GetSurface (refedges.Get(1).surfnr2);
+
+      len = curvelength.Last();
+      ne = int (len + 0.5);
+      if (ne == 0) ne = 1;
+      if (Dist2 (edgepoints[1], edgepoints.Last()) < 1e-8 && 
+      ne <= 6) 
+      ne = 6;
+      corr = len / ne;
+    */
+
+    // generate initial point
+    Point<3> p = edgepoints[0];
+    PointIndex pi1 = -1;
+    for (pi = PointIndex::BASE; 
+	 pi < mesh.GetNP()+PointIndex::BASE; pi++)
+
+      if (Dist (mesh[pi], p) < 1e-6)
+	{
+	  pi1 = pi;
+	  break;
+	}
+
+    if (pi1 == -1) pi1 = mesh.AddPoint (p, layer);
+
+    p = edgepoints.Last();
+    PointIndex pi2 = -1;
+    for (pi = PointIndex::BASE; 
+	 pi < mesh.GetNP()+PointIndex::BASE; pi++)
+
+      if (Dist (mesh[pi], p) < 1e-6)
+	{
+	  pi2 = pi;
+	  break;
+	}
+    if (pi2==-1) pi2 = mesh.AddPoint (p, layer);
+
+    /*
+  
+    j = 1;
+    for (i = 1; i <= ne; i++)
+    {
+    while (curvelength[j] < i * corr && j < curvelength.Size()) j++;
+      
+    lam = (i * corr - curvelength[j-1]) / 
+    (curvelength[j] - curvelength[j-1]);
+      
+    np(0) = (1-lam) * edgepoints[j-1](0) + lam * edgepoints[j](0);
+    np(1) = (1-lam) * edgepoints[j-1](1) + lam * edgepoints[j](1);
+    np(2) = (1-lam) * edgepoints[j-1](2) + lam * edgepoints[j](2);
+      
+      
+    thispi = 0;
+    if (i == ne)
+    for (j = 1; j <= mesh.GetNP(); j++)
+    if (Dist(mesh.Point(j), np) < 1e-6)
+    thispi = j;
+      
+    if (!thispi)
+    {
+    ProjectToEdge (surf1, surf2, np);
+    thispi = mesh.AddPoint (np);
+    }
+    */
+  
+    for (k = 1; k <= refedges.Size(); k++)
+      {
+	if (refedgesinv.Get(k))
+	  {
+	    seg.p1 = pi1;
+	    seg.p2 = pi2;
+	  }
+	else
+	  {
+	    seg.p1 = pi2;
+	    seg.p2 = pi1;
+	  }
+
+	seg.si = refedges.Get(k).si;
+	seg.domin = refedges.Get(k).domin;
+	seg.domout = refedges.Get(k).domout;
+	seg.tlosurf = refedges.Get(k).tlosurf;
+	seg.edgenr = refedges.Get(k).edgenr;
+	seg.surfnr1 = refedges.Get(k).surfnr1;
+	seg.surfnr2 = refedges.Get(k).surfnr2;
+	seg.seginfo = 0;
+	if (k == 1) seg.seginfo = (refedgesinv.Get(k)) ? 2 : 1;
+	mesh.AddSegment (seg);
+	//	  (*testout) << "add seg " << seg.p1 << "-" << seg.p2 << endl;
+      }
+  }
+  
+
+
+
+
+
+
+  void EdgeCalculation :: 
+  CopyEdge (const ARRAY<Segment> & refedges,
+	    const ARRAY<int> & refedgesinv,
+	    int copyfromedge, 
+	    const Point<3> & fromstart, const Point<3> & fromend,
+	    const Point<3> & tostart, const Point<3> & toend,
+	    int copyedgeidentification, 
+	    int layer,
+	    Mesh & mesh)
+  {
+    int i, j, k;
+    PointIndex pi;
+
+    // copy start and end points
+    for (i = 1; i <= 2; i++)
+      {
+	Point<3> fromp =
+	  (i == 1) ? fromstart : fromend;
+	Point<3> top =
+	  (i == 1) ? tostart : toend;
+      
+	PointIndex frompi = -1;
+	PointIndex topi = -1;
+	for (pi = PointIndex::BASE; 
+	     pi < mesh.GetNP()+PointIndex::BASE; pi++)
+	  {
+	    if (Dist2 (mesh[pi], fromp) <= 1e-16)
+	      frompi = pi;
+	    if (Dist2 (mesh[pi], top) <= 1e-16)
+	      topi = pi;
+	  }
+
+	if (topi == -1)
+	  topi = mesh.AddPoint (top, layer);
+
+	const Identification & csi = 
+	  (*geometry.identifications.Get(copyedgeidentification));
+
+	if (csi.Identifyable (mesh[frompi], mesh[topi]))
+	  mesh.GetIdentifications().Add(frompi, topi, copyedgeidentification);
+	else if (csi.Identifyable (mesh[topi], mesh[frompi]))
+	  mesh.GetIdentifications().Add(topi, frompi, copyedgeidentification);
+	else
+	  {
+	    cerr << "edgeflw.cpp: should identify, but cannot";
+	    exit(1);
+	  }
+	/*
+	  (*testout) << "Add Identification from CopyEdge, p1 = " 
+	  << mesh[PointIndex(frompi)] << ", p2 = " 
+	  << mesh[PointIndex(topi)] << endl;
+
+	  mesh.GetIdentifications().Add(frompi, topi, copyedgeidentification);
+	*/
+      }
+
+    int oldns = mesh.GetNSeg();
+    for (i = 1; i <= oldns; i++)
+      {
+	// real copy, since array might be reallocated !!
+	const Segment oldseg = mesh.LineSegment(i);
+	if (oldseg.edgenr != copyfromedge)
+	  continue;
+	if (oldseg.seginfo == 0)
+	  continue;
+
+	int pi1 = oldseg.p1;
+	int pi2 = oldseg.p2;
+
+	int npi1 = geometry.identifications.Get(copyedgeidentification)
+	  -> GetIdentifiedPoint (mesh, pi1);
+	int npi2 = geometry.identifications.Get(copyedgeidentification)
+	  -> GetIdentifiedPoint (mesh, pi2);
+
+	Segment seg;
+
+	for (k = 1; k <= refedges.Size(); k++)
+	  {
+	    int inv = refedgesinv.Get(k);
+
+	    // other edge is inverse
+	    if (oldseg.seginfo == 1)
+	      inv = !inv;
+
+	    //	  (*testout) << "inv, now = " << inv << endl;
+
+	    if (inv)
+	      {
+		seg.p1 = npi1;
+		seg.p2 = npi2;
+	      }
+	    else
+	      {
+		seg.p1 = npi2;
+		seg.p2 = npi1;
+	      }
+	    seg.si = refedges.Get(k).si;
+	    seg.domin = refedges.Get(k).domin;
+	    seg.domout = refedges.Get(k).domout;
+	    seg.tlosurf = refedges.Get(k).tlosurf;
+	    seg.edgenr = refedges.Get(k).edgenr;
+	    seg.surfnr1 = refedges.Get(k).surfnr1;
+	    seg.surfnr2 = refedges.Get(k).surfnr2;
+	    seg.seginfo = 0;
+	    if (k == 1) seg.seginfo = refedgesinv.Get(k) ? 2 : 1;
+	    mesh.AddSegment (seg);
+	    //	  (*testout) << "copy seg " << seg.p1 << "-" << seg.p2 << endl;
+#ifdef DEVELOP
+
+	    (*testout) << "copy seg, face = " << seg.si << ": " 
+		       << " inv = " << inv << ", refinv = " << refedgesinv.Get(k)
+		       << mesh.Point(seg.p1) << ", " << mesh.Point(seg.p2) << endl;
+#endif
+
+	  }
+      
+      }   
+  }
+  
+
+
+
+
+
+
+  void EdgeCalculation :: 
+  FindClosedSurfaces (double h, Mesh & mesh)
+  {
+    // if there is no special point at a sphere, one has to add a segment pair
+  
+    int i, j; 
+    int nsol; 
+    int nsurf = geometry.GetNSurf();
+    int layer;
+
+    BitArray pointatsurface (nsurf);
+    Point<3> p1, p2;
+    Vec<3> nv, tv;
+    Solid * tansol;
+    ARRAY<int> tansurfind;
+    //  const Solid * sol;
+
+    nsol = geometry.GetNTopLevelObjects();
+
+
+    pointatsurface.Clear();
+  
+    /*
+      for (i = 1; i <= specpoints.Size(); i++)
+      {
+      int classrep;
+
+      classrep = geometry.GetSurfaceClassRepresentant (specpoints[i].s1);
+      pointatsurface.Set (classrep);
+      classrep = geometry.GetSurfaceClassRepresentant (specpoints[i].s2);
+      pointatsurface.Set (classrep);
+      //      pointatsurface.Set (specpoints[i].s1);
+      //      pointatsurface.Set (specpoints[i].s2);
+      }
+    */
+    for (i = 1; i <= mesh.GetNSeg(); i++)
+      {
+	const Segment & seg = mesh.LineSegment(i);
+	int classrep;
+
+#ifdef DEVELOP      
+	(*testout) << seg.surfnr1 << ", " << seg.surfnr2 << ", si = " << seg.si << endl;
+#endif
+	classrep = geometry.GetSurfaceClassRepresentant (seg.si);
+
+	pointatsurface.Set (classrep);
+      }
+
+  
+    for (i = 0; i < nsurf; i++)
+      {
+	int classrep = geometry.GetSurfaceClassRepresentant (i);
+
+	if (!pointatsurface.Test(classrep))
+	  {
+	    const Surface * s = geometry.GetSurface(i);
+	    p1 = s -> GetSurfacePoint();
+	    s -> GetNormalVector (p1, nv);
+		    
+	    double hloc = 
+	      min2 (s->LocH (p1, 3, 1, h), mesh.GetH(p1));
+
+	    tv = nv.GetNormal ();
+	    tv *=  (hloc / tv.Length());
+	    p2 = p1 + tv;
+	    s->Project (p2);
+	  
+		    
+	    Segment seg1;
+	    seg1.si = i;
+	    seg1.domin = -1;
+	    seg1.domout = -1;
+
+	    Segment seg2;
+	    seg2.si = i;
+	    seg2.domin = -1;
+	    seg2.domout = -1;
+
+	    seg1.surfnr1 = i;
+	    seg2.surfnr1 = i;
+	    seg1.surfnr2 = i;
+	    seg2.surfnr2 = i;
+
+	    for (j = 0; j < nsol; j++)
+	      {
+		if (geometry.GetTopLevelObject(j)->GetSurface())
+		  continue;
+
+		const Solid * sol = geometry.GetTopLevelObject(j)->GetSolid();
+		sol -> TangentialSolid (p1, tansol);
+		layer = geometry.GetTopLevelObject(j)->GetLayer();
+
+		if (tansol)
+		  {
+		    tansol -> GetSurfaceIndices (tansurfind);
+		
+		    if (tansurfind.Size() == 1 && tansurfind.Get(1) == i)
+		      {
+			if (!tansol->VectorIn(p1, nv))
+			  {
+			    seg1.domin = j;
+			    seg2.domin = j;
+			    seg1.tlosurf = j;
+			    seg2.tlosurf = j;
+			  }
+			else
+			  {
+			    seg1.domout = j;
+			    seg2.domout = j;
+			    seg1.tlosurf = j;
+			    seg2.tlosurf = j;
+			  }
+			//        seg.s2 = i;
+			//        seg.invs1 = surfaces[i] -> Inverse();
+			//        seg.invs2 = ! (surfaces[i] -> Inverse());
+		      }
+		    delete tansol;
+		  }
+	      }
+
+
+	    if (seg1.domin != -1 || seg1.domout != -1)
+	      {
+		mesh.AddPoint (p1, layer);
+		mesh.AddPoint (p2, layer);
+		seg1.p1 = mesh.GetNP()-1;
+		seg1.p2 = mesh.GetNP();
+		seg2.p2 = mesh.GetNP()-1;
+		seg2.p1 = mesh.GetNP();
+		seg1.geominfo[0].trignum = 1;
+		seg1.geominfo[1].trignum = 1;
+		seg2.geominfo[0].trignum = 1;
+		seg2.geominfo[1].trignum = 1;
+		mesh.AddSegment (seg1);
+		mesh.AddSegment (seg2);
+
+		PrintMessage (5, "Add line segment to smooth surface");
+
+#ifdef DEVELOP
+		(*testout) << "Add segment at smooth surface " << i;
+		if (i != classrep) (*testout) << ", classrep = " << classrep;
+		(*testout) << ": "
+			   << mesh.Point (mesh.GetNP()-1) << " - "
+			   << mesh.Point (mesh.GetNP()) << endl;
+#endif
+	      }
+	  }
+      }
+  }
+
+}
diff --git a/Netgen/libsrc/csg/genmesh.cpp b/Netgen/libsrc/csg/genmesh.cpp
index 62a336e260..c052624a42 100644
--- a/Netgen/libsrc/csg/genmesh.cpp
+++ b/Netgen/libsrc/csg/genmesh.cpp
@@ -1,709 +1,684 @@
 #include <mystdlib.h>
 
 
-// #include <FlexLexer.h>
-
 #include <myadt.hpp>
 
 #include <linalg.hpp>
 #include <csg.hpp>
-#include <stlgeom.hpp>
 #include <meshing.hpp>
 
+
 namespace netgen
 {
-ARRAY<SpecialPoint> specpoints;
-static ARRAY<MeshPoint> spoints;
+  ARRAY<SpecialPoint> specpoints;
+  static ARRAY<MeshPoint> spoints;
 
 #define TCL_OK 0
 #define TCL_ERROR 1
 
 
-static void FindPoints (CSGeometry & geom, Mesh & mesh,
-			const char * filename)
-{
-  int i;
-  PrintMessage (1, "Start Findpoints");
 
-  char * savetask = multithread.task;
-  multithread.task = "Find points";
+  static void FindPoints (CSGeometry & geom, Mesh & mesh)
+  {
+    PrintMessage (1, "Start Findpoints");
 
-  for (i = 0; i < geom.GetNUserPoints(); i++)
-    {
-      mesh.AddPoint (geom.GetUserPoint (i));
-      mesh.AddLockedPoint (PointIndex (i+1));
-    }
+    char * savetask = multithread.task;
+    multithread.task = "Find points";
 
-  SpecialPointCalculation spc;
+    for (int i = 0; i < geom.GetNUserPoints(); i++)
+      {
+	mesh.AddPoint (geom.GetUserPoint (i));
+	mesh.AddLockedPoint (PointIndex (i+1));
+      }
 
-  if (spoints.Size() == 0)
-    spc.CalcSpecialPoints (geom, spoints);
+    SpecialPointCalculation spc;
 
-  PrintMessage (2, "Analyze spec points");
-  spc.AnalyzeSpecialPoints (geom, spoints, specpoints);
+    if (spoints.Size() == 0)
+      spc.CalcSpecialPoints (geom, spoints);
+    
+    PrintMessage (2, "Analyze spec points");
+    spc.AnalyzeSpecialPoints (geom, spoints, specpoints);
   
-  PrintMessage (5, "done");
+    PrintMessage (5, "done");
+
+    (*testout) << specpoints.Size() << " special points:" << endl;
+    for (int i = 0; i < specpoints.Size(); i++)
+      specpoints[i].Print (*testout);
+
+    /*
+      for (int i = 1; i <= geom.identifications.Size(); i++)
+      geom.identifications.Elem(i)->IdentifySpecialPoints (specpoints);
+    */
+    multithread.task = savetask;
+  }
+
+
+
+
+
+
+  static void FindEdges (CSGeometry & geom, Mesh & mesh)
+  {
+    EdgeCalculation ec (geom, specpoints);
+    ec.Calc (mparam.maxh, mesh);
+
+    for (int i = 0; i < geom.singedges.Size(); i++)
+      geom.singedges[i]->FindPointsOnEdge (mesh);
+    for (int i = 0; i < geom.singpoints.Size(); i++)
+      geom.singpoints[i]->FindPoints (mesh);
+
+    for (int i = 1; i <= mesh.GetNSeg(); i++)
+      {
+	int ok = 0;
+	for (int k = 1; k <= mesh.GetNFD(); k++)
+	  if (mesh.GetFaceDescriptor(k).SegmentFits (mesh.LineSegment(i)))
+	    ok = k;
+
+	if (!ok)
+	  ok = mesh.AddFaceDescriptor (FaceDescriptor (mesh.LineSegment(i)));
+
+	mesh.LineSegment(i).si = ok;
+      }
+
+    for (int i = 0; i < geom.identifications.Size(); i++)
+      geom.identifications[i]->IdentifyPoints (mesh);
+    for (int i = 0; i < geom.identifications.Size(); i++)
+      geom.identifications[i]->IdentifyFaces (mesh);
+
+
+
+    // find intersecting segments
+    PrintMessage (3, "Check intersecting edges");
+
+    if (!ec.point_on_edge_problem)
+      for (SegmentIndex si = 0; si < mesh.GetNSeg(); si++)
+	for (SegmentIndex sj = 0; sj < si; sj++)
+	  {
+	    if (!mesh[si].seginfo || !mesh[sj].seginfo) continue;
+	    if (mesh[mesh[si].p1].GetLayer() != mesh[mesh[sj].p2].GetLayer()) continue;
+	    
+	    Point<3> pi1 = mesh[mesh[si].p1];
+	    Point<3> pi2 = mesh[mesh[si].p2];
+	    Point<3> pj1 = mesh[mesh[sj].p1];
+	    Point<3> pj2 = mesh[mesh[sj].p2];
+	    Vec<3> vi = pi2 - pi1;
+	    Vec<3> vj = pj2 - pj1;
+	    
+	    if (sqr (vi * vj) > (1-1e-6) * Abs2 (vi) * Abs2 (vj)) continue;
+
+	    // pi1 + vi t = pj1 + vj s
+	    Mat<3,2> mat;
+	    Vec<3> rhs;
+	    Vec<2> sol;
+	    
+	    for (int j = 0; j < 3; j++)
+	      { 
+		mat(j,0) = vi(j); 
+		mat(j,1) = -vj(j); 
+		rhs(j) = pj1(j)-pi1(j); 
+	    }
+	    
+	    mat.Solve (rhs, sol);
 
-  (*testout) << specpoints.Size() << " special points:" << endl;
-  for (i = 0; i < specpoints.Size(); i++)
-    specpoints[i].Print (*testout);
+	    if (sol(0) > 1e-6 && sol(0) < 1-1e-6 &&
+		sol(1) > 1e-6 && sol(1) < 1-1e-6 &&
+		Abs (rhs - mat*sol) < 1e-6)
+	      {
+		Point<3> ip = pi1 + sol(0) * vi;
+		cout << "Intersection at " << ip << endl;
+		
+		geom.AddUserPoint (ip);
+		spoints.Append (MeshPoint (ip, mesh[mesh[si].p1].GetLayer()));
+		mesh.AddPoint (ip);
+	      }
+	  }
+  }  
 
-  /*
-  for (i = 1; i <= geom.identifications.Size(); i++)
-    geom.identifications.Elem(i)->IdentifySpecialPoints (specpoints);
-  */
-  multithread.task = savetask;
-}
 
 
 
-static void FindEdges (CSGeometry & geom, Mesh & mesh,
-		       const char * filename)
-{
-  int i, k;
+
+
+  static void MeshSurface (CSGeometry & geom, Mesh & mesh)
+  {
+    char * savetask = multithread.task;
+    multithread.task = "Surface meshing";
   
-  EdgeCalculation ec (geom, specpoints);
-  ec.Calc (mparam.maxh, mesh);
+    ARRAY<Segment> segments;
+    int noldp = mesh.GetNP();
 
-  for (i = 1; i <= geom.singedges.Size(); i++)
-    geom.singedges.Elem(i)->FindPointsOnEdge (mesh);
-  for (i = 1; i <= geom.singpoints.Size(); i++)
-    geom.singpoints.Elem(i)->FindPoints (mesh);
+    double starttime = GetTime();
 
-  //  SaveEdges (mesh, filename, h, "edges.out");
+    // find master faces from identified
+    ARRAY<int> masterface(mesh.GetNFD());
+    for (int i = 1; i <= mesh.GetNFD(); i++)
+      masterface.Elem(i) = i;
+  
+    ARRAY<INDEX_2> fpairs;
+    bool changed;
+    do
+      {
+	changed = 0;
+	for (int i = 0; i < geom.identifications.Size(); i++)
+	  {
+	    geom.identifications[i]->GetIdentifiedFaces (fpairs);
+
+	    for (int j = 0; j < fpairs.Size(); j++)
+	      {
+		if (masterface.Get(fpairs[j].I1()) <
+		    masterface.Get(fpairs[j].I2()))
+		  {
+		    changed = 1;
+		    masterface.Elem(fpairs[j].I2()) =
+		      masterface.Elem(fpairs[j].I1());
+		  }
+		if (masterface.Get(fpairs[j].I2()) <
+		    masterface.Get(fpairs[j].I1()))
+		  {
+		    changed = 1;
+		    masterface.Elem(fpairs[j].I1()) =
+		      masterface.Elem(fpairs[j].I2());
+		  }
+	      }
+	  }
+      }
+    while (changed);
+
+
+    int bccnt=0;
+    for (int k = 0; k < geom.GetNSurf(); k++)
+      bccnt = max2 (bccnt, geom.GetSurface(k)->GetBCProperty());
+
+    for (int k = 1; k <= mesh.GetNFD(); k++)
+      {
+	FaceDescriptor & fd = mesh.GetFaceDescriptor(k);
+	const Surface * surf = geom.GetSurface(fd.SurfNr());
+
+	if (fd.TLOSurface() && 
+	    geom.GetTopLevelObject(fd.TLOSurface()-1) -> GetBCProp() > 0)
+	  fd.SetBCProperty (geom.GetTopLevelObject(fd.TLOSurface()-1) -> GetBCProp());
+	else if (surf -> GetBCProperty() != -1)
+	  fd.SetBCProperty (surf->GetBCProperty());
+	else
+	  {
+	    bccnt++;
+	    fd.SetBCProperty (bccnt);
+	  }      
+
+	for (int l = 0; l < geom.bcmodifications.Size(); l++)
+	  {
+	    if (geom.GetSurfaceClassRepresentant (fd.SurfNr()) == 
+		geom.GetSurfaceClassRepresentant (geom.bcmodifications[l].si) &&
+		(fd.DomainIn() == geom.bcmodifications[l].tlonr+1 ||
+		 fd.DomainOut() == geom.bcmodifications[l].tlonr+1))
+	      {
+		fd.SetBCProperty (geom.bcmodifications[l].bcnr);
+	      }
+	  }
+      }
+
+
+    for (int j = 0; j < geom.singfaces.Size(); j++)
+      {
+	ARRAY<int> surfs;
+	geom.GetIndependentSurfaceIndices (geom.singfaces[j]->GetSolid(),
+					   geom.BoundingBox(), surfs);
+	for (int k = 1; k <= mesh.GetNFD(); k++)
+	  {
+	    FaceDescriptor & fd = mesh.GetFaceDescriptor(k);
+	    for (int l = 0; l < surfs.Size(); l++)
+	      if (surfs[l] == fd.SurfNr())
+		{
+		  if (geom.singfaces[j]->GetDomainNr() == fd.DomainIn())
+		    fd.domin_singular = 1;
+		  if (geom.singfaces[j]->GetDomainNr() == fd.DomainOut())
+		    fd.domout_singular = 1;
+		}
+	  }
+      }
+    
 
-  for (i = 1; i <= mesh.GetNSeg(); i++)
-    {
-      int ok = 0;
-      for (k = 1; k <= mesh.GetNFD(); k++)
-	if (mesh.GetFaceDescriptor(k).SegmentFits (mesh.LineSegment(i)))
-	  ok = k;
+    // assemble edge hash-table
+    mesh.CalcSurfacesOfNode();
 
-      if (!ok)
-	ok = mesh.AddFaceDescriptor (FaceDescriptor (mesh.LineSegment(i)));
+    for (int k = 1; k <= mesh.GetNFD(); k++)
+      {
+	multithread.percent = 100.0 * k / (mesh.GetNFD()+1e-10);
 
-      mesh.LineSegment(i).si = ok;
-    }
+	if (masterface.Get(k) != k)
+	  continue;
 
-  (*testout) << "identify points after line meshing" << endl;
-  for (i = 0; i < geom.identifications.Size(); i++)
-    geom.identifications[i]->IdentifyPoints (mesh);
-  for (i = 0; i < geom.identifications.Size(); i++)
-    geom.identifications[i]->IdentifyFaces (mesh);
+	FaceDescriptor & fd = mesh.GetFaceDescriptor(k);
 
-  //  mesh.Save ("lines.vol");
-}  
+	(*testout) << "Surface " << k << endl;
+	(*testout) << "Face Descriptor: " << fd << endl;
+	PrintMessage (1, "Surface ", k, " / ", mesh.GetNFD());
 
+	int oldnf = mesh.GetNSE();
+      
+	const Surface * surf =
+	  geom.GetSurface((mesh.GetFaceDescriptor(k).SurfNr()));
 
-static void MeshSurface (CSGeometry & geom, Mesh & mesh)
-{
-  int i, j, k;
-  int changed;
 
-  char * savetask = multithread.task;
-  multithread.task = "Surface meshing";
+	Meshing2Surfaces meshing(*surf, geom.BoundingBox());
+	meshing.SetStartTime (starttime);
 
+	for (PointIndex pi = PointIndex::BASE; pi < noldp+PointIndex::BASE; pi++)
+	  meshing.AddPoint (mesh[pi], pi);
   
-  ARRAY<Segment> segments;
-  int noldp = mesh.GetNP();
+	segments.SetSize (0);
+
+	for (SegmentIndex si = 0; si < mesh.GetNSeg(); si++)
+	  if (mesh[si].si == k)
+	    segments.Append (mesh[si]);
+
+	for (int i = 1; i <= geom.identifications.Size(); i++)
+	  geom.identifications.Get(i)->
+	    BuildSurfaceElements(segments, mesh, surf);
+
+	for (int si = 0; si < segments.Size(); si++)
+	  {
+	    PointGeomInfo gi;
+	    gi.trignum = k;
+	    meshing.AddBoundaryElement (segments[si].p1 + 1 - PointIndex::BASE, 
+					segments[si].p2 + 1 - PointIndex::BASE, 
+					gi, gi);
+	  }
+
+	double maxh = mparam.maxh;
+	if (fd.DomainIn() != 0)
+	  {
+	    const Solid * s1 = 
+	      geom.GetTopLevelObject(fd.DomainIn()-1) -> GetSolid();
+	    if (s1->GetMaxH() < maxh)
+	      maxh = s1->GetMaxH();
+	    maxh = min2(maxh, geom.GetTopLevelObject(fd.DomainIn()-1)->GetMaxH());
+	  }
+	if (fd.DomainOut() != 0)
+	  {
+	    const Solid * s1 = 
+	      geom.GetTopLevelObject(fd.DomainOut()-1) -> GetSolid();
+	    if (s1->GetMaxH() < maxh)
+	      maxh = s1->GetMaxH();
+	    maxh = min2(maxh, geom.GetTopLevelObject(fd.DomainOut()-1)->GetMaxH());
+	  }
+	if (fd.TLOSurface() != 0)
+	  {
+	    double hi = geom.GetTopLevelObject(fd.TLOSurface()-1) -> GetMaxH();
+	    if (hi < maxh) maxh = hi;
+	  }
+
+	(*testout) << "domin = " << fd.DomainIn() << ", domout = " << fd.DomainOut()
+		   << ", tlo-surf = " << fd.TLOSurface()
+		   << " mpram.maxh = " << mparam.maxh << ", maxh = " << maxh << endl;
+
+	mparam.checkoverlap = 0;
+
+	MESHING2_RESULT res =
+	  meshing.GenerateMesh (mesh, maxh, k);
+
+	if (res != MESHING2_OK)
+	  {
+	    PrintError ("Problem in Surface mesh generation");
+	    throw NgException ("Problem in Surface mesh generation");
+	  }
+
+	if (multithread.terminate) return;
+      
+	for (int i = oldnf+1; i <= mesh.GetNSE(); i++)
+	  mesh.SurfaceElement(i).SetIndex (k);
 
-  double starttime = GetTime();
 
+	//      mesh.CalcSurfacesOfNode();
+	if (segments.Size())   
+	  { 
+	    // surface was meshed, not copied
+	    PrintMessage (2, "Optimize Surface");
+	    for (int i = 1; i <= mparam.optsteps2d; i++)
+	      {
+		if (multithread.terminate) return;
 
-  // find master faces from identified
-  ARRAY<int> masterface(mesh.GetNFD());
-  for (i = 1; i <= mesh.GetNFD(); i++)
-    masterface.Elem(i) = i;
-  
-  ARRAY<INDEX_2> fpairs;
-  do
-    {
-      changed = 0;
-      for (i = 0; i < geom.identifications.Size(); i++)
-	{
-	  geom.identifications[i]->GetIdentifiedFaces (fpairs);
-
-	  for (j = 0; j < fpairs.Size(); j++)
-	    {
-	      if (masterface.Get(fpairs[j].I1()) <
-		  masterface.Get(fpairs[j].I2()))
 		{
-		  changed = 1;
-		  masterface.Elem(fpairs[j].I2()) =
-		    masterface.Elem(fpairs[j].I1());
+		  MeshOptimize2dSurfaces meshopt(geom);
+		  meshopt.SetFaceIndex (k);
+		  meshopt.SetImproveEdges (0);
+		  meshopt.SetMetricWeight (0.2);
+		  meshopt.SetWriteStatus (0);
+
+		  meshopt.EdgeSwapping (mesh, (i > mparam.optsteps2d/2));
 		}
-	      if (masterface.Get(fpairs[j].I2()) <
-		  masterface.Get(fpairs[j].I1()))
+
+		if (multithread.terminate) return;
 		{
-		  changed = 1;
-		  masterface.Elem(fpairs[j].I1()) =
-		    masterface.Elem(fpairs[j].I2());
-		}
-	    }
-	}
-    }
-  while (changed);
-
-
-  int bccnt=0;
-  for (k = 0; k < geom.GetNSurf(); k++)
-    bccnt = max2 (bccnt, geom.GetSurface(k)->GetBCProperty());
-
-  for (k = 1; k <= mesh.GetNFD(); k++)
-    {
-      FaceDescriptor & fd = mesh.GetFaceDescriptor(k);
-      const Surface * surf = geom.GetSurface(fd.SurfNr());
-
-      if (fd.TLOSurface() && 
-	  geom.GetTopLevelObject(fd.TLOSurface()-1) -> GetBCProp() > 0)
-	fd.SetBCProperty (geom.GetTopLevelObject(fd.TLOSurface()-1) -> GetBCProp());
-      else if (surf -> GetBCProperty() != -1)
-	fd.SetBCProperty (surf->GetBCProperty());
-      else
-	{
-	  bccnt++;
-	  fd.SetBCProperty (bccnt);
-	}      
-
-      for (int l = 0; l < geom.bcmodifications.Size(); l++)
-	{
-	  if (geom.GetSurfaceClassRepresentant (fd.SurfNr()) == 
-	      geom.GetSurfaceClassRepresentant (geom.bcmodifications[l].si) &&
-	      (fd.DomainIn() == geom.bcmodifications[l].tlonr+1 ||
-	       fd.DomainOut() == geom.bcmodifications[l].tlonr+1))
-	    {
-	      cout << "modify, surfnr = " << fd.SurfNr() << ", dom = " 
-		   << geom.bcmodifications[l].tlonr+1 << endl;
-	      fd.SetBCProperty (geom.bcmodifications[l].bcnr);
-	    }
-	}
-    }
+		  //		mesh.CalcSurfacesOfNode();
+		
+		  MeshOptimize2dSurfaces meshopt(geom);
+		  meshopt.SetFaceIndex (k);
+		  meshopt.SetImproveEdges (0);
+		  meshopt.SetMetricWeight (0.2);
+		  meshopt.SetWriteStatus (0);
 
-  /*
-  (*testout) << "Face descriptors:" << endl;
-  for (k = 1; k <= mesh.GetNFD(); k++)
-    (*testout) << "fd(" << k << ") = " << mesh.GetFaceDescriptor(k) << endl;
-  */
+		  meshopt.ImproveMesh (mesh);
+		}
 
-  // assemble edge hash-table
-  (*testout) << "calc surf before surface" << endl;
-  mesh.CalcSurfacesOfNode();
+		{
+		  MeshOptimize2dSurfaces meshopt(geom);
+		  meshopt.SetFaceIndex (k);
+		  meshopt.SetImproveEdges (0);
+		  meshopt.SetMetricWeight (0.2);
+		  meshopt.SetWriteStatus (0);
+
+		  meshopt.CombineImprove (mesh);
+		  //		mesh.CalcSurfacesOfNode();
+		}
 
-  for (k = 1; k <= mesh.GetNFD(); k++)
-    {
-      (*testout) << "mesh face " << k << endl;
-      multithread.percent = 100 * k / (mesh.GetNFD()+1e-10);
+		if (multithread.terminate) return;
+		{
+		  MeshOptimize2dSurfaces meshopt(geom);
+		  meshopt.SetFaceIndex (k);
+		  meshopt.SetImproveEdges (0);
+		  meshopt.SetMetricWeight (0.2);
+		  meshopt.SetWriteStatus (0);
 
-      if (masterface.Get(k) != k)
-	continue;
+		  meshopt.ImproveMesh (mesh);
+		}
+	      }
+	  }
 
-      FaceDescriptor & fd = mesh.GetFaceDescriptor(k);
 
-      (*testout) << "Surface " << k << endl;
-      (*testout) << "Face Descriptor: " << fd << endl;
-      PrintMessage (1, "Surface ", k, " / ", mesh.GetNFD());
+	PrintMessage (3, (mesh.GetNSE() - oldnf), " elements, ", mesh.GetNP(), " points");
 
-      int oldnf = mesh.GetNSE();
-      
-      const Surface * surf =
-	geom.GetSurface((mesh.GetFaceDescriptor(k).SurfNr()));
-
-      /*
-      if (surf -> GetBCProperty() != -1)
-	fd.SetBCProperty (surf->GetBCProperty());
-      else
-	{
-	  bccnt++;
-	  fd.SetBCProperty (bccnt);
-	}
-      */
-
-
-      Meshing2Surfaces meshing(*surf, geom.BoundingBox());
-      meshing.SetStartTime (starttime);
-
-      for (PointIndex pi = PointIndex::BASE; 
-	   pi < noldp+PointIndex::BASE; pi++)
-	{
-	  meshing.AddPoint (mesh[pi], pi);
-	}
-  
-      segments.SetSize (0);
-
-      for (i = 1; i <= mesh.GetNSeg(); i++)
-	{
-	  Segment * seg = &mesh.LineSegment(i);
-	  if (seg->si == k)
-	    {
-	      segments.Append (*seg);
-	      (*testout) << "add segment " << seg->p1 << "-" << seg->p2 << endl;
-	    }
-	}
-
-      for (i = 1; i <= geom.identifications.Size(); i++)
-	geom.identifications.Get(i)->
-	  BuildSurfaceElements(segments, mesh, surf);
-
-      // (*testout) << "Mesh surface, elements:" << endl;
-      for (SegmentIndex si = 0; si < segments.Size(); si++)
-	{
-	  PointGeomInfo gi;
-	  gi.trignum = k;
-	  /*
-	  (*testout) << segments[si].p1 << " - " << segments[si].p2 << endl;
-	  (*testout) << mesh[segments[si].p1] << " - " 
-		     << mesh[segments[si].p2] << endl;
-	  */
-	  meshing.AddBoundaryElement (segments[si].p1 + 1 - PointIndex::BASE, 
-				      segments[si].p2 + 1 - PointIndex::BASE, 
-				      gi, gi);
-	}
-
-      double maxh = mparam.maxh;
-      if (fd.DomainIn() != 0)
-	{
-	  const Solid * s1 = 
-	    geom.GetTopLevelObject(fd.DomainIn()-1) -> GetSolid();
-	  if (s1->GetMaxH() < maxh)
-	    maxh = s1->GetMaxH();
-	  maxh = min2(maxh, geom.GetTopLevelObject(fd.DomainIn()-1)->GetMaxH());
-	}
-      if (fd.DomainOut() != 0)
-	{
-	  const Solid * s1 = 
-	    geom.GetTopLevelObject(fd.DomainOut()-1) -> GetSolid();
-	  if (s1->GetMaxH() < maxh)
-	    maxh = s1->GetMaxH();
-	  maxh = min2(maxh, geom.GetTopLevelObject(fd.DomainOut()-1)->GetMaxH());
-	}
-      if (fd.TLOSurface() != 0)
-	{
-	  double hi = geom.GetTopLevelObject(fd.TLOSurface()-1) -> GetMaxH();
-	  if (hi < maxh) maxh = hi;
-	}
-
-      (*testout) << "domin = " << fd.DomainIn() << ", domout = " << fd.DomainOut()
-		 << ", tlo-surf = " << fd.TLOSurface()
-		 << " mpram.maxh = " << mparam.maxh << ", maxh = " << maxh << endl;
-
-
-      MESHING2_RESULT res =
-	meshing.GenerateMesh (mesh, maxh, k);
-
-      if (res != MESHING2_OK)
-	{
-	  PrintError ("Problem in Surface mesh generation");
-	  throw NgException ("Problem in Surface mesh generation");
-	}
-      if (multithread.terminate) return;
+#ifdef OPENGL
+	extern void Render();
+	Render();
+#endif
+      }
+    
+    mesh.Compress();
+
+    do
+      {
+	changed = 0;
+	for (int k = 1; k <= mesh.GetNFD(); k++)
+	  {
+	    multithread.percent = 100.0 * k / (mesh.GetNFD()+1e-10);
+	  
+	    if (masterface.Get(k) == k)
+	      continue;
 
-      
-      for (i = oldnf+1; i <= mesh.GetNSE(); i++)
-	mesh.SurfaceElement(i).SetIndex (k);
+	    FaceDescriptor & fd = mesh.GetFaceDescriptor(k);
 
+	    (*testout) << "Surface " << k << endl;
+	    (*testout) << "Face Descriptor: " << fd << endl;
+	    PrintMessage (2, "Surface ", k);
 
-      //      mesh.CalcSurfacesOfNode();
-      if (segments.Size())   
-	{ 
-	  // surface was meshed, not copied
-	  PrintMessage (2, "Optimize Surface");
-	  for (i = 1; i <= mparam.optsteps2d; i++)
-	    {
-	      if (multithread.terminate) return;
+	    int oldnf = mesh.GetNSE();
+      
+	    const Surface * surf =
+	      geom.GetSurface((mesh.GetFaceDescriptor(k).SurfNr()));
 
+	    /*
+	      if (surf -> GetBCProperty() != -1)
+	      fd.SetBCProperty (surf->GetBCProperty());
+	      else
 	      {
-		MeshOptimize2dSurfaces meshopt(geom);
-		meshopt.SetFaceIndex (k);
-		meshopt.SetImproveEdges (0);
-		meshopt.SetMetricWeight (0.2);
-		meshopt.SetWriteStatus (0);
-
-		meshopt.EdgeSwapping (mesh, (i > mparam.optsteps2d/2));
+	      bccnt++;
+	      fd.SetBCProperty (bccnt);
 	      }
-
-	      if (multithread.terminate) return;
+	    */
+  
+	    segments.SetSize (0);
+	    for (int i = 1; i <= mesh.GetNSeg(); i++)
 	      {
-		//		mesh.CalcSurfacesOfNode();
-		
-		MeshOptimize2dSurfaces meshopt(geom);
-		meshopt.SetFaceIndex (k);
-		meshopt.SetImproveEdges (0);
-		meshopt.SetMetricWeight (0.2);
-		meshopt.SetWriteStatus (0);
-
-		meshopt.ImproveMesh (mesh);
+		Segment * seg = &mesh.LineSegment(i);
+		if (seg->si == k)
+		  segments.Append (*seg);
 	      }
 
+	    for (int i = 1; i <= geom.identifications.Size(); i++)
 	      {
-		MeshOptimize2dSurfaces meshopt(geom);
-		meshopt.SetFaceIndex (k);
-		meshopt.SetImproveEdges (0);
-		meshopt.SetMetricWeight (0.2);
-		meshopt.SetWriteStatus (0);
-
-		meshopt.CombineImprove (mesh);
-		//		mesh.CalcSurfacesOfNode();
+		geom.identifications.Elem(i)->GetIdentifiedFaces (fpairs);
+		int found = 0;
+		for (int j = 1; j <= fpairs.Size(); j++)
+		  if (fpairs.Get(j).I1() == k || fpairs.Get(j).I2() == k)
+		    found = 1;
+
+		if (!found)
+		  continue;
+
+		geom.identifications.Get(i)->
+		  BuildSurfaceElements(segments, mesh, surf);
+		if (!segments.Size())
+		  break;
 	      }
 
-	      if (multithread.terminate) return;
-	      {
-		MeshOptimize2dSurfaces meshopt(geom);
-		meshopt.SetFaceIndex (k);
-		meshopt.SetImproveEdges (0);
-		meshopt.SetMetricWeight (0.2);
-		meshopt.SetWriteStatus (0);
-
-		meshopt.ImproveMesh (mesh);
-	      }
-	    }
-	}
+	  
+	    if (multithread.terminate) return;
 
+	    for (int i = oldnf+1; i <= mesh.GetNSE(); i++)
+	      mesh.SurfaceElement(i).SetIndex (k);
 
 
-      PrintMessage (3, (mesh.GetNSE() - oldnf), " elements, ", mesh.GetNP(), " points");
+	    if (!segments.Size())
+	      {
+		masterface.Elem(k) = k;
+		changed = 1; 
+	      }
 
+	    PrintMessage (3, (mesh.GetNSE() - oldnf), " elements, ", mesh.GetNP(), " points");
+	  }
+      
 #ifdef OPENGL
-      extern void Render();
-      Render();
+	extern void Render();
+	Render();
 #endif
-    }
+      }
+    while (changed);
 
-  mesh.Compress();
+    mesh.SplitSeparatedFaces();
+    mesh.CalcSurfacesOfNode();
 
+    multithread.task = savetask;
+  }
 
 
 
 
 
 
-  do
-    {
-      changed = 0;
-      for (k = 1; k <= mesh.GetNFD(); k++)
-	{
-	  multithread.percent = 100 * k / (mesh.GetNFD()+1e-10);
-	  
-	  if (masterface.Get(k) == k)
-	    continue;
 
-	  FaceDescriptor & fd = mesh.GetFaceDescriptor(k);
+  int GenerateMesh (CSGeometry & geom,
+		    Mesh *& mesh,
+		    int perfstepsstart, int perfstepsend,
+		    const char * optstr)
+  {
 
-	  (*testout) << "Surface " << k << endl;
-	  (*testout) << "Face Descriptor: " << fd << endl;
-	  PrintMessage (2, "Surface ", k);
+    if (mesh && mesh->GetNSE() &&
+	!geom.GetNSolids())
+      {
+	if (perfstepsstart < MESHCONST_MESHVOLUME)
+	  perfstepsstart = MESHCONST_MESHVOLUME;
+      }
 
-	  int oldnf = mesh.GetNSE();
-      
-	  const Surface * surf =
-	    geom.GetSurface((mesh.GetFaceDescriptor(k).SurfNr()));
-
-	  /*
-	  if (surf -> GetBCProperty() != -1)
-	    fd.SetBCProperty (surf->GetBCProperty());
-	  else
-	    {
-	      bccnt++;
-	      fd.SetBCProperty (bccnt);
-	    }
-	  */
-  
-	  segments.SetSize (0);
-	  for (i = 1; i <= mesh.GetNSeg(); i++)
-	    {
-	      Segment * seg = &mesh.LineSegment(i);
-	      if (seg->si == k)
-		segments.Append (*seg);
-	    }
-
-	  for (i = 1; i <= geom.identifications.Size(); i++)
-	    {
-	      geom.identifications.Elem(i)->GetIdentifiedFaces (fpairs);
-	      int found = 0;
-	      for (j = 1; j <= fpairs.Size(); j++)
-		if (fpairs.Get(j).I1() == k || fpairs.Get(j).I2() == k)
-		  found = 1;
-
-	      if (!found)
-		continue;
-
-	      geom.identifications.Get(i)->
-		BuildSurfaceElements(segments, mesh, surf);
-	      if (!segments.Size())
-		break;
-	    }
 
-	  
-	  if (multithread.terminate) return;
 
-	  for (i = oldnf+1; i <= mesh.GetNSE(); i++)
-	    mesh.SurfaceElement(i).SetIndex (k);
+    if (perfstepsstart <= MESHCONST_ANALYSE)
+      {
+	delete mesh;
+	mesh = new Mesh();
 
+	mesh->SetGlobalH (mparam.maxh);
 
-	  if (!segments.Size())
-	    {
-	      masterface.Elem(k) = k;
-	      changed = 1; 
-	    }
+	ARRAY<double> maxhdom(geom.GetNTopLevelObjects());
+	for (int i = 0; i < maxhdom.Size(); i++)
+	  maxhdom[i] = geom.GetTopLevelObject(i)->GetMaxH();
 
-	  PrintMessage (3, (mesh.GetNSE() - oldnf), " elements, ", mesh.GetNP(), " points");
-	}
-      
-#ifdef OPENGL
-      extern void Render();
-      Render();
-#endif
-    }
-  while (changed);
+	mesh->SetMaxHDomain (maxhdom);
 
-  mesh.SplitSeparatedFaces();
+	if (mparam.uselocalh)
+	  {
+	    double maxsize = geom.MaxSize(); 
+	    mesh->SetLocalH (Point<3>(-maxsize, -maxsize, -maxsize),
+			     Point<3>(maxsize, maxsize, maxsize),
+			     mparam.grading);
 
-  mesh.CalcSurfacesOfNode();
+	    mesh -> LoadLocalMeshSize (mparam.meshsizefilename);
+	  }
 
-  multithread.task = savetask;
-}
-
-
-int GenerateMesh (CSGeometry & geom,
-		  Mesh *& mesh,
-		  int perfstepsstart, int perfstepsend,
-		  const char * optstr)
-{
-  extern char geomfilename[];
-  const char * filename = geomfilename;
-
-  int i;
-  char filenametrunc[255];
-  char filenameext[255];
-
-  if (mesh && mesh->GetNSE() &&
-      !geom.GetNSolids())
-    {
-      if (perfstepsstart < MESHCONST_MESHVOLUME)
-	perfstepsstart = MESHCONST_MESHVOLUME;
-    }
-
-
-  
-  strcpy (filenametrunc, filename);
-  char * pos = strstr (filenametrunc, ".geo");
-  if (pos) strcpy (pos, "");
-
-  if (perfstepsstart <= MESHCONST_ANALYSE)
-    {
-      delete mesh;
-      mesh = new Mesh();
-
-      mesh->SetGlobalH (mparam.maxh);
-
-      ARRAY<double> maxhdom(geom.GetNTopLevelObjects());
-      for (i = 0; i < maxhdom.Size(); i++)
-	maxhdom[i] = geom.GetTopLevelObject(i)->GetMaxH();
-
-      mesh->SetMaxHDomain (maxhdom);
-
-      if (mparam.uselocalh)
-	{
-	  //	  double maxsize = globflags.GetNumFlag ("maxsize", 500);
-	  double maxsize = geom.MaxSize(); 
-	  mesh->SetLocalH (Point<3>(-maxsize, -maxsize, -maxsize),
-			   Point<3>(maxsize, maxsize, maxsize),
-			   mparam.grading);
-
-	  if (mparam.meshsizefilename)
-	    {
-	      ifstream msf(mparam.meshsizefilename);
-	      if (msf)
-		mesh->LoadLocalMeshSize (msf);
-	    }
-	}
-
-      spoints.SetSize(0);
-      FindPoints (geom, *mesh, filename);
+	spoints.SetSize(0);
+	FindPoints (geom, *mesh);
       
-      PrintMessage (5, "find points done");
+	PrintMessage (5, "find points done");
 
 #ifdef LOG_STREAM
-      (*logout) << "Special points found" << endl
-		<< "time = " << GetTime() << " sec" << endl
-		<< "points: " << mesh->GetNP() << endl << endl;
+	(*logout) << "Special points found" << endl
+		  << "time = " << GetTime() << " sec" << endl
+		  << "points: " << mesh->GetNP() << endl << endl;
 #endif
-    }
+      }
 
 
-  if (multithread.terminate || perfstepsend <= MESHCONST_ANALYSE) 
-    return TCL_OK;
+    if (multithread.terminate || perfstepsend <= MESHCONST_ANALYSE) 
+      return TCL_OK;
 
-  if (perfstepsstart <= MESHCONST_MESHEDGES)
-    {
-      FindEdges (geom, *mesh, filename);
-      if (multithread.terminate) return TCL_OK;
+
+    if (perfstepsstart <= MESHCONST_MESHEDGES)
+      {
+	FindEdges (geom, *mesh);
+	if (multithread.terminate) return TCL_OK;
 #ifdef LOG_STREAM      
-      (*logout) << "Edges meshed" << endl
-		<< "time = " << GetTime() << " sec" << endl
-		<< "points: " << mesh->GetNP() << endl;
+	(*logout) << "Edges meshed" << endl
+		  << "time = " << GetTime() << " sec" << endl
+		  << "points: " << mesh->GetNP() << endl;
 #endif
       
       
-      if (multithread.terminate)
-	return TCL_OK;
+	if (multithread.terminate)
+	  return TCL_OK;
   
-
-      if (mparam.uselocalh)
-	{
-	  mesh->CalcLocalH();
-	  mesh->DeleteMesh();
+	if (mparam.uselocalh)
+	  {
+	    mesh->CalcLocalH();
+	    mesh->DeleteMesh();
 	  
-	  FindPoints (geom, *mesh, filename);
-	  if (multithread.terminate) return TCL_OK;
-	  FindEdges (geom, *mesh, filename);
-	  if (multithread.terminate) return TCL_OK;
+	    FindPoints (geom, *mesh);
+	    if (multithread.terminate) return TCL_OK;
+	    FindEdges (geom, *mesh);
+	    if (multithread.terminate) return TCL_OK;
 
-	  mesh->DeleteMesh();
+	    mesh->DeleteMesh();
 	  
-	  FindPoints (geom, *mesh, filename);
-	  if (multithread.terminate) return TCL_OK;
-	  FindEdges (geom, *mesh, filename);
-	  if (multithread.terminate) return TCL_OK;
-	}
-    }
+	    FindPoints (geom, *mesh);
+	    if (multithread.terminate) return TCL_OK;
+	    FindEdges (geom, *mesh);
+	    if (multithread.terminate) return TCL_OK;
+	  }
+      }
   
-  if (multithread.terminate || perfstepsend <= MESHCONST_MESHEDGES)
-    return TCL_OK;
+    if (multithread.terminate || perfstepsend <= MESHCONST_MESHEDGES)
+      return TCL_OK;
 
 
-  if (perfstepsstart <= MESHCONST_MESHSURFACE)
-    {
-      MeshSurface (geom, *mesh);  
-      if (multithread.terminate) return TCL_OK;
+    if (perfstepsstart <= MESHCONST_MESHSURFACE)
+      {
+	MeshSurface (geom, *mesh);  
+	if (multithread.terminate) return TCL_OK;
       
 #ifdef LOG_STREAM
-      (*logout) << "Surfaces meshed" << endl
-		<< "time = " << GetTime() << " sec" << endl
-		<< "points: " << mesh->GetNP() << endl;
+	(*logout) << "Surfaces meshed" << endl
+		  << "time = " << GetTime() << " sec" << endl
+		  << "points: " << mesh->GetNP() << endl;
 #endif      
       
-      if (mparam.uselocalh && 0)
-	{
-	  mesh->CalcLocalH();      
-	  mesh->DeleteMesh();
+	if (mparam.uselocalh && 0)
+	  {
+	    mesh->CalcLocalH();      
+	    mesh->DeleteMesh();
 
-	  FindPoints (geom, *mesh, filename);
-	  if (multithread.terminate) return TCL_OK;
-	  FindEdges (geom, *mesh, filename);
-	  if (multithread.terminate) return TCL_OK;
+	    FindPoints (geom, *mesh);
+	    if (multithread.terminate) return TCL_OK;
+	    FindEdges (geom, *mesh);
+	    if (multithread.terminate) return TCL_OK;
 
-	  MeshSurface (geom, *mesh);  
-	  if (multithread.terminate) return TCL_OK;
-	}
+	    MeshSurface (geom, *mesh);  
+	    if (multithread.terminate) return TCL_OK;
+	  }
 
 #ifdef LOG_STREAM      
-      (*logout) << "Surfaces remeshed" << endl
-		<< "time = " << GetTime() << " sec" << endl
-		<< "points: " << mesh->GetNP() << endl;
+	(*logout) << "Surfaces remeshed" << endl
+		  << "time = " << GetTime() << " sec" << endl
+		  << "points: " << mesh->GetNP() << endl;
 #endif      
       
 #ifdef STAT_STREAM
-      (*statout) << mesh->GetNSeg() << " & "
-		 << mesh->GetNSE() << " & - &" 
-		 << GetTime() << " & " << endl;
+	(*statout) << mesh->GetNSeg() << " & "
+		   << mesh->GetNSE() << " & - &" 
+		   << GetTime() << " & " << endl;
 #endif  
 
-      MeshQuality2d (*mesh);
-      mesh->CalcSurfacesOfNode();
-
-      /*
-	strcpy (filenameext, filenametrunc);
-	strcat (filenameext, ".surf.mesh");
-	WriteFile (WRITE_SURFACE, mesh, geom, filenameext, filename, h);
-      */
-    }
+	MeshQuality2d (*mesh);
+	mesh->CalcSurfacesOfNode();
+      }
   
-  if (multithread.terminate || perfstepsend <= MESHCONST_OPTSURFACE)
-    return TCL_OK;
+    if (multithread.terminate || perfstepsend <= MESHCONST_OPTSURFACE)
+      return TCL_OK;
 
 
-  if (perfstepsstart <= MESHCONST_MESHVOLUME)
-    {
-      multithread.task = "Volume meshing";
-      
-      MESHING3_RESULT res =
-	MeshVolume (mparam, *mesh);
+    if (perfstepsstart <= MESHCONST_MESHVOLUME)
+      {
+	multithread.task = "Volume meshing";
 
-      if (res != MESHING3_OK) return TCL_ERROR;
+	MESHING3_RESULT res =
+	  MeshVolume (mparam, *mesh);
+
+	if (res != MESHING3_OK) return TCL_ERROR;
       
-      if (multithread.terminate) return TCL_OK;
+	if (multithread.terminate) return TCL_OK;
       
-      RemoveIllegalElements (*mesh);
-      if (multithread.terminate) return TCL_OK;
+	RemoveIllegalElements (*mesh);
+	if (multithread.terminate) return TCL_OK;
 
-      MeshQuality3d (*mesh);
+	MeshQuality3d (*mesh);
       
-      for (int i = 0; i < geom.GetNTopLevelObjects(); i++)
-	mesh->SetMaterial (i+1, geom.GetTopLevelObject(i)->GetMaterial().c_str());
+	for (int i = 0; i < geom.GetNTopLevelObjects(); i++)
+	  mesh->SetMaterial (i+1, geom.GetTopLevelObject(i)->GetMaterial().c_str());
       
 
 #ifdef STAT_STREAM
-      (*statout) << GetTime() << " & ";
+	(*statout) << GetTime() << " & ";
 #endif      
       
 #ifdef LOG_STREAM
-      (*logout) << "Volume meshed" << endl
-		<< "time = " << GetTime() << " sec" << endl
-	    << "points: " << mesh->GetNP() << endl;
+	(*logout) << "Volume meshed" << endl
+		  << "time = " << GetTime() << " sec" << endl
+		  << "points: " << mesh->GetNP() << endl;
 #endif
-    }
+      }
 
-  if (multithread.terminate || perfstepsend <= MESHCONST_MESHVOLUME)
-    return TCL_OK;
+    if (multithread.terminate || perfstepsend <= MESHCONST_MESHVOLUME)
+      return TCL_OK;
 
 
-  if (perfstepsstart <= MESHCONST_OPTVOLUME)
-    {
-      multithread.task = "Volume optimization";
+    if (perfstepsstart <= MESHCONST_OPTVOLUME)
+      {
+	multithread.task = "Volume optimization";
       
-      OptimizeVolume (mparam, *mesh, &geom);
-      if (multithread.terminate) return TCL_OK;
+	OptimizeVolume (mparam, *mesh);
+	if (multithread.terminate) return TCL_OK;
       
 #ifdef STAT_STREAM
-      (*statout) << GetTime() << " & "
-		 << mesh->GetNE() << " & "
-		 << mesh->GetNP() << " " << '\\' << '\\' << " \\" << "hline" << endl;
+	(*statout) << GetTime() << " & "
+		   << mesh->GetNE() << " & "
+		   << mesh->GetNP() << " " << '\\' << '\\' << " \\" << "hline" << endl;
 #endif      
 
 #ifdef LOG_STREAM      
-      (*logout) << "Volume optimized" << endl
-		<< "time = " << GetTime() << " sec" << endl
-	    << "points: " << mesh->GetNP() << endl;
+	(*logout) << "Volume optimized" << endl
+		  << "time = " << GetTime() << " sec" << endl
+		  << "points: " << mesh->GetNP() << endl;
 #endif
+      }
 
-      
-#ifndef TRAFO
-      /*
-      if (&geom)
-	{
-	  ZRefinementOptions opt;
-	  opt.minref = 0;
-	  ZRefinement (*mesh, &geom, opt);
-	}
-      */
-#endif
-    }
-
-  /*
-  cout << (char)7 << (char)7 << endl;
-  cout << "Meshing finished" << endl << endl;
-  cout << setw (4) << mesh->GetNP() << " Points" << endl;
-  cout << setw (4) << mesh->GetNSE() << " Surface Elements" << endl;
-  cout << setw (4) << mesh->GetNE() << " Volume Elements" << endl;
-  cout << setw (4) << int (GetTime()) << " Seconds" << endl;
-  */
-
-  /*
-    strcpy (filenameext, filenametrunc);
-    strcat (filenameext, ".vol");
-    mesh->Save (filenameext);	
-  */
-
-  return TCL_OK;
-}
+    return TCL_OK;
+  }
 }
diff --git a/Netgen/libsrc/csg/identify.cpp b/Netgen/libsrc/csg/identify.cpp
index 337bc26a4a..4622ee42bb 100644
--- a/Netgen/libsrc/csg/identify.cpp
+++ b/Netgen/libsrc/csg/identify.cpp
@@ -208,7 +208,7 @@ Identifyable (const SpecialPoint & sp1, const SpecialPoint & sp2) const
 	continue;
 
       Vec<3> n1;
-      s1->GetNormalVector (hsp1.p, n1);
+      n1 = s1->GetNormalVector (hsp1.p);
       n1 /= n1.Length();
       if ( fabs(n1 * hsp1.v) > 1e-3)
 	continue;
@@ -218,7 +218,7 @@ Identifyable (const SpecialPoint & sp1, const SpecialPoint & sp2) const
 	continue;
 
       Vec<3> n2;
-      s2->GetNormalVector (hsp2.p, n2);
+      n2 = s2->GetNormalVector (hsp2.p);
       n2 /= n2.Length();
       if ( fabs(n2 * hsp2.v) > 1e-3)
 	continue;
@@ -475,7 +475,7 @@ BuildSurfaceElements (ARRAY<Segment> & segs,
 				 Point<3> (mesh.Point (newel.PNum(1))));
 	      
 	      Vec<3> nsurf;
-	      geom.GetSurface (surfnr)->GetNormalVector (mesh.Point(newel.PNum(1)), nsurf);
+	      nsurf = geom.GetSurface (surfnr)->GetNormalVector (mesh.Point(newel.PNum(1)));
 	      if (nsurf * nt < 0)
 		Swap (newel.PNum(2), newel.PNum(3));
 				
@@ -525,17 +525,21 @@ CloseSurfaceIdentification (int anr,
 			    const CSGeometry & ageom,
 			    const Surface * as1,
 			    const Surface * as2,
+			    const TopLevelObject * adomain,
 			    const Flags & flags)
   : Identification(anr, ageom)
 {
   s1 = as1;
   s2 = as2;
+  domain = adomain;
   ref_levels = int (flags.GetNumFlag ("reflevels", 2));
   ref_levels_s1 = int (flags.GetNumFlag ("reflevels1", 0));
   ref_levels_s2 = int (flags.GetNumFlag ("reflevels2", 0));
   slices = flags.GetNumListFlag ("slices");
   // eps_n = 1e-3;
   eps_n = 1e-6;
+
+  dom_surf_valid = 0;
 }
 
 CloseSurfaceIdentification :: ~CloseSurfaceIdentification ()
@@ -575,7 +579,7 @@ void CloseSurfaceIdentification :: IdentifySpecialPoints
       if (!s1->PointOnSurface (p1))
 	continue;
 
-      s1->GetNormalVector (p1, n1);
+	s1->GetNormalVector (p1, n1);
       n1 /= n1.Length();
       if ( fabs(n1 * points.Get(i).v) > 1e-3)
 	continue;
@@ -622,65 +626,71 @@ void CloseSurfaceIdentification :: IdentifySpecialPoints
 int CloseSurfaceIdentification :: 
 Identifyable (const SpecialPoint & sp1, const SpecialPoint & sp2) const
 {
-  int i;
-  double val;
 
-  for (i = 1; i <= 1; i++)
+  if (!dom_surf_valid)
     {
-      if (!s1->PointOnSurface (sp1.p))
-	continue;
+      const_cast<bool&> (dom_surf_valid) = 1;
+      ARRAY<int> & hsurf = const_cast<ARRAY<int>&> (domain_surfaces);
 
-      Vec<3> n1;
-      s1->GetNormalVector (sp1.p, n1);
-      n1.Normalize();
-      if ( fabs(n1 * sp1.v) > eps_n)
-	continue;
+      if (domain)
+	{
+	  BoxSphere<3> hbox (geom.BoundingBox());
+	  geom.GetIndependentSurfaceIndices (domain->GetSolid(), hbox,
+					     hsurf);
+	}
+      else
+	{
+	  hsurf.SetSize (geom.GetNSurf());
+	  for (int j = 0; j < hsurf.Size(); j++)
+	    hsurf[j] = j;
+	}
+    }
 
-      if (!s2->PointOnSurface(sp2.p))
-	continue;
 
-      Vec<3> n2;
-      s2->GetNormalVector (sp2.p, n2);
-      n2.Normalize();
-      if ( fabs(n2 * sp2.v) > eps_n)
-	continue;
 
-      // must have joint surface 
-      bool joint = 0;
-      for (int j = 0; j < geom.GetNSurf(); j++)
+  if (!s1->PointOnSurface (sp1.p))
+    return 0;
+  
+  Vec<3> n1 = s1->GetNormalVector (sp1.p);
+  n1.Normalize();
+  if ( fabs(n1 * sp1.v) > eps_n)
+    return 0;
+  
+  if (!s2->PointOnSurface(sp2.p))
+    return 0;
+  
+  Vec<3> n2 = s2->GetNormalVector (sp2.p);
+  n2.Normalize();
+  if ( fabs(n2 * sp2.v) > eps_n)
+    return 0;
+  
+  // must have joint surface 
+  bool joint = 0;
+  //  for (int j = 0; j < geom.GetNSurf(); j++)
+  for (int jj = 0; jj < domain_surfaces.Size(); jj++)
+    {
+      int j = domain_surfaces[jj];
+      if (geom.GetSurface(j) -> PointOnSurface(sp1.p) &&
+	  geom.GetSurface(j) -> PointOnSurface(sp2.p) )
 	{
-	  if (geom.GetSurface(j) -> PointOnSurface(sp1.p) &&
-	      geom.GetSurface(j) -> PointOnSurface(sp2.p) )
+	  Vec<3> hn1 = geom.GetSurface(j)->GetNormalVector (sp1.p);
+	  Vec<3> hn2 = geom.GetSurface(j)->GetNormalVector (sp2.p);
+	  
+	  if (hn1 * hn2 > 0)
 	    {
-	      Vec<3> hn1, hn2;
-	      geom.GetSurface(j)->GetNormalVector (sp1.p, hn1);
-	      geom.GetSurface(j)->GetNormalVector (sp2.p, hn2);
-
-	      if (hn1 * hn2 > 0)
-		{
-		  joint = 1;
-		  break;
-		}
+	      joint = 1;
+	      break;
 	    }
 	}
-      if (!joint) continue;
-
-      Vec<3> v = sp2.p - sp1.p;
-      double vl = v.Length();
-      double cl = fabs (v*n1);
-      
-      /*
-      val = 1 - cl*cl/(vl*vl);
-      val += (sp1.v - sp2.v).Length();
-    
-      if (val < 1e-3)
-	{
-	  return 1;
-	}
-      */
-      if (cl > (1-eps_n*eps_n) * vl && (sp1.v - sp2.v).Length() < 0.1)
-	return 1;
     }
+  if (!joint) return 0;
+  
+  Vec<3> v = sp2.p - sp1.p;
+  double vl = v.Length();
+  double cl = fabs (v*n1);
+      
+  if (cl > (1-eps_n*eps_n) * vl && (sp1.v - sp2.v).Length() < 0.1)
+    return 1;
 
   return 0;
 }
@@ -688,8 +698,7 @@ Identifyable (const SpecialPoint & sp1, const SpecialPoint & sp2) const
 int CloseSurfaceIdentification :: 
 Identifyable (const Point<3> & p1, const Point<3> & p2) const
 {
-  return (s1->PointOnSurface (p1) &&
-	  s2->PointOnSurface (p2));
+  return (s1->PointOnSurface (p1) && s2->PointOnSurface (p2));
 }
   
 
@@ -701,7 +710,7 @@ IdentifyableCandidate (const SpecialPoint & sp1) const
   if (s1->PointOnSurface (sp1.p))
     {
       Vec<3> n1;
-      s1->GetNormalVector (sp1.p, n1);
+      n1 = s1->GetNormalVector (sp1.p);
       n1.Normalize();
       if ( fabs(n1 * sp1.v) > eps_n)
 	return 0;
@@ -711,7 +720,7 @@ IdentifyableCandidate (const SpecialPoint & sp1) const
   if (s2->PointOnSurface (sp1.p))
     {
       Vec<3> n1;
-      s2->GetNormalVector (sp1.p, n1);
+      n1 = s2->GetNormalVector (sp1.p);
       n1.Normalize();
       if ( fabs(n1 * sp1.v) > eps_n)
 	return 0;
@@ -725,14 +734,8 @@ IdentifyableCandidate (const SpecialPoint & sp1) const
 int CloseSurfaceIdentification :: 
 ShortEdge (const SpecialPoint & sp1, const SpecialPoint & sp2) const
 {
-  int i;
-  double val;
-  
-  SpecialPoint hsp1 = sp1;
-  SpecialPoint hsp2 = sp2;
-
-  if ( (s1->PointOnSurface (hsp1.p) && s2->PointOnSurface (hsp2.p)) ||
-       (s1->PointOnSurface (hsp2.p) && s2->PointOnSurface (hsp1.p)) )
+  if ( (s1->PointOnSurface (sp1.p) && s2->PointOnSurface (sp2.p)) ||
+       (s1->PointOnSurface (sp2.p) && s2->PointOnSurface (sp1.p)) )
     {
       return 1;
     }
@@ -833,7 +836,7 @@ void CloseSurfaceIdentification :: IdentifyPoints (Mesh & mesh)
 	  double mindist = 1e10;
 
 	  Vec<3> n1;
-	  s1->GetNormalVector (p1, n1);
+	  n1 = s1->GetNormalVector (p1);
 	  n1.Normalize();
 
 	  for (i2 = 1; i2 <= np; i2++)
@@ -852,6 +855,28 @@ void CloseSurfaceIdentification :: IdentifyPoints (Mesh & mesh)
 	      Vec<3> n = p2 - p1;
 	      n.Normalize();
 	      
+	      
+	      bool joint = 0;
+	      for (int jj = 0; jj < domain_surfaces.Size(); jj++)
+		{
+		  int j = domain_surfaces[jj];
+		  if (geom.GetSurface(j) -> PointOnSurface(p1) &&
+		      geom.GetSurface(j) -> PointOnSurface(p2) )
+		    {
+		      Vec<3> hn1 = geom.GetSurface(j)->GetNormalVector (p1);
+		      Vec<3> hn2 = geom.GetSurface(j)->GetNormalVector (p2);
+		      
+		      if (hn1 * hn2 > 0)
+			{
+			  joint = 1;
+			  break;
+			}
+		    }
+		}
+	      if (!joint) continue;
+	      
+
+
 
 	      if (fabs (n * n1) > 0.9 &&
 		  Dist (p1, p2) < mindist)
@@ -1026,7 +1051,7 @@ BuildSurfaceElements (ARRAY<Segment> & segs,
 			      Point<3> (mesh.Point(el.PNum(1))));
 
 	    Vec<3> ns;
-	    surf->GetNormalVector (mesh.Point(el.PNum(1)), ns);
+	    ns = surf->GetNormalVector (mesh.Point(el.PNum(1)));
 	    // (*testout) << "n = " << n << " ns = " << ns << endl;
 	    if (n * ns < 0)
 	      {
@@ -1102,7 +1127,7 @@ BuildSurfaceElements (ARRAY<Segment> & segs,
 		  Vec<3> nt = Cross (Vec<3> (mesh.Point (newel.PNum(1)), mesh.Point (newel.PNum(2))),
 				    Vec<3> (mesh.Point (newel.PNum(1)), mesh.Point (newel.PNum(3))));
 		  Vec<3> nsurf;
-		  geom.GetSurface (surfnr)->GetNormalVector (mesh.Point(newel.PNum(1)), nsurf);
+		  nsurf = geom.GetSurface (surfnr)->GetNormalVector (mesh.Point(newel.PNum(1)));
 		  if (nsurf * nt < 0)
 		    Swap (newel.PNum(2), newel.PNum(3));
 		  
@@ -1183,7 +1208,7 @@ BuildSurfaceElements2 (ARRAY<Segment> & segs,
 				 Point<3> (mesh.Point (newel.PNum(3)))- 
 				 Point<3> (mesh.Point (newel.PNum(1))));
 	      Vec<3> nsurf;
-	      geom.GetSurface (surfnr)->GetNormalVector (mesh.Point(newel.PNum(1)), nsurf);
+	      nsurf = geom.GetSurface (surfnr)->GetNormalVector (mesh.Point(newel.PNum(1)));
 	      if (nsurf * nt < 0)
 		Swap (newel.PNum(2), newel.PNum(3));
 	      
@@ -1288,7 +1313,7 @@ void CloseEdgesIdentification :: IdentifySpecialPoints
       if (!s1->PointOnSurface (p1))
 	continue;
 
-      s1->GetNormalVector (p1, n1);
+	s1->GetNormalVector (p1, n1);
       n1 /= n1.Length();
       if ( fabs(n1 * points.Get(i).v) > 1e-3)
 	continue;
@@ -1347,7 +1372,7 @@ Identifyable (const SpecialPoint & sp1, const SpecialPoint & sp2) const
 	continue;
 
       Vec<3> n1;
-      s1->GetNormalVector (hsp1.p, n1);
+      n1 = s1->GetNormalVector (hsp1.p);
       n1 /= n1.Length();
       if ( fabs(n1 * hsp1.v) > 1e-3)
 	continue;
@@ -1357,7 +1382,7 @@ Identifyable (const SpecialPoint & sp1, const SpecialPoint & sp2) const
 	continue;
 
       Vec<3> n2;
-      s2->GetNormalVector (hsp2.p, n2);
+      n2 = s2->GetNormalVector (hsp2.p);
       n2 /= n2.Length();
       if ( fabs(n2 * hsp2.v) > 1e-3)
 	continue;
@@ -1412,8 +1437,8 @@ void CloseEdgesIdentification :: IdentifyPoints (Mesh & mesh)
 	Vec<3> n = p2 - p1;
 	n.Normalize();
 
-	s1->GetNormalVector (p1, n1);
-	facet->GetNormalVector (p1, nf);
+	n1 = s1->GetNormalVector (p1);
+	nf = facet->GetNormalVector (p1);
 	t = Cross (n1, nf);
 	t /= t.Length();
 
@@ -1455,7 +1480,7 @@ BuildSurfaceElements (ARRAY<Segment> & segs,
 			      Point<3> (mesh.Point(el.PNum(3)))-
 			      Point<3> (mesh.Point(el.PNum(1))));
 	    Vec<3> ns;
-	    surf->GetNormalVector (mesh.Point(el.PNum(1)), ns);
+	    ns = surf->GetNormalVector (mesh.Point(el.PNum(1)));
 	    (*testout) << "n = " << n << " ns = " << ns << endl;
 	    if (n * ns < 0)
 	      {
diff --git a/Netgen/libsrc/csg/identify.hpp b/Netgen/libsrc/csg/identify.hpp
index b358a424c4..16e371b200 100644
--- a/Netgen/libsrc/csg/identify.hpp
+++ b/Netgen/libsrc/csg/identify.hpp
@@ -95,10 +95,12 @@ public:
 
 
 ///
+class TopLevelObject;
 class CloseSurfaceIdentification : public Identification
 {
   const Surface * s1;
   const Surface * s2;
+  const TopLevelObject * domain;
   /// number of refinement levels (in Z-refinement)
   int ref_levels;
   /// number of refinement levels for layer next to s1 (in Z-refinement)
@@ -108,11 +110,16 @@ class CloseSurfaceIdentification : public Identification
   ///
   double eps_n;
   ARRAY<double> slices;
+  /// used only for domain-local identification:
+  ARRAY<int> domain_surfaces;
+  ///
+  bool dom_surf_valid;
 public:
   CloseSurfaceIdentification (int anr, 
 			      const CSGeometry & ageom,
 			      const Surface * as1,
 			      const Surface * as2,
+			      const TopLevelObject * adomain,
 			      const Flags & flags);
   virtual ~CloseSurfaceIdentification ();
 
diff --git a/Netgen/libsrc/csg/polyhedra.cpp b/Netgen/libsrc/csg/polyhedra.cpp
index 9a6855227f..f0ce2f63f5 100644
--- a/Netgen/libsrc/csg/polyhedra.cpp
+++ b/Netgen/libsrc/csg/polyhedra.cpp
@@ -23,18 +23,16 @@ Polyhedra::Face::Face (int pi1, int pi2, int pi3, const ARRAY<Point<3> > & point
   nn = n;
   nn.Normalize();
   //  PseudoInverse (v1, v2, w1, w2);
-
   
   Mat<2,3> mat;
   Mat<3,2> inv;
-  int i, j;
-  for (i = 0; i < 3; i++)
+  for (int i = 0; i < 3; i++)
     {
       mat(0,i) = v1(i);
       mat(1,i) = v2(i);
     }
   CalcInverse (mat, inv);
-  for (i = 0; i < 3; i++)
+  for (int i = 0; i < 3; i++)
     {
       w1(i) = inv(i,0);
       w2(i) = inv(i,1);
@@ -79,10 +77,7 @@ INSOLID_TYPE Polyhedra :: BoxInSolid (const BoxSphere<3> & box) const
 
       double dist2 = MinDistTP2 (p1, p2, p3, box.Center());
       if (dist2 < sqr (box.Diam()/2))
-	{
-	  //	  (*testout) << "intersect" << endl;
-	  return DOES_INTERSECT;
-	}
+	return DOES_INTERSECT;
     };
 
   return PointInSolid (box.Center(), 1e-3 * box.Diam());
@@ -92,10 +87,9 @@ INSOLID_TYPE Polyhedra :: BoxInSolid (const BoxSphere<3> & box) const
 INSOLID_TYPE Polyhedra :: PointInSolid (const Point<3> & p,
 					double eps) const
 {
-  //  (*testout) << "Point in Sol, p = " << p << " " << flush;
   Vec<3> n, v1, v2;
 
-  // "random numbers":
+  // random (?) numbers:
   n(0) = 0.123871;
   n(1) = 0.15432;
   n(2) = 0.43989;
@@ -124,13 +118,10 @@ INSOLID_TYPE Polyhedra :: PointInSolid (const Point<3> & p,
       if (lam3 < eps)
 	{
 	  if (lam1 >= -eps && lam2 >= -eps && lam1+lam2 <= 1+eps)
-	    {
-	      return DOES_INTERSECT;
-	    }
+	    return DOES_INTERSECT;
 	}
       else if (lam1 >= 0 && lam2 >= 0 && lam1+lam2 <= 1)
-	//	  lam3 > 0)
-	{
+	{  // lam3 > 0
 	  cnt++;
 	}
 
@@ -146,10 +137,100 @@ INSOLID_TYPE Polyhedra :: VecInSolid (const Point<3> & p,
 				      const Vec<3> & v,
 				      double eps) const
 {
+  int point_on_n_faces = 0;
+  INSOLID_TYPE res;
+
+  Vec<3> vn = v;
+  vn.Normalize();
+  for (int i = 0; i < faces.Size(); i++)
+    {
+      const Point<3> & p1 = points[faces[i].pnums[0]];
+      
+      Vec<3> v0 = p - p1;
+      double lam3 = -(faces[i].n * v0);
+
+      if (fabs (lam3) > eps) continue;
+
+      double lam1 = (faces[i].w1 * v0);
+      double lam2 = (faces[i].w2 * v0);
+
+      if (lam1 >= -eps && lam2 >= -eps && lam1+lam2 <= 1+eps)
+	{
+	  point_on_n_faces++;
+
+	  double scal = vn * faces[i].n;
+	
+	  res = DOES_INTERSECT;
+	  if (scal > eps) res = IS_OUTSIDE;
+	  if (scal < -eps) res = IS_INSIDE;
+	}
+    }
+
+  if (point_on_n_faces == 1)
+    return res;
+
+  
   Point<3> p2 = p + (1e-3/(v.Length()+1e-16)) * v;
-  return PointInSolid (p2, eps);
+  res = PointInSolid (p2, eps);
+  //  (*testout) << "p = " << p << " v = " << v << " p2 = " << p2 << endl;
+  //  (*testout) << "polyeder::vecinsolid = " << int(res) << endl;
+  return res;
+}
+
+
+INSOLID_TYPE Polyhedra :: VecInSolid2 (const Point<3> & p,
+				       const Vec<3> & v1,
+				       const Vec<3> & v2,
+				       double eps) const
+{
+  int point_on_n_faces = 0;
+  INSOLID_TYPE res;
+
+  Vec<3> v1n = v1;
+  v1n.Normalize();
+  Vec<3> v2n = v2;
+  v2n.Normalize();
+
+
+  for (int i = 0; i < faces.Size(); i++)
+    {
+      const Point<3> & p1 = points[faces[i].pnums[0]];
+      
+      Vec<3> v0 = p - p1;
+      double lam3 = -(faces[i].n * v0);
+
+      if (fabs (lam3) > eps) continue;
+
+      double lam1 = (faces[i].w1 * v0);
+      double lam2 = (faces[i].w2 * v0);
+
+      if (lam1 >= -eps && lam2 >= -eps && lam1+lam2 <= 1+eps)
+	{
+	  double scal1 = v1n * faces[i].n;
+	  if (fabs (scal1) > eps) continue;
+
+
+	  point_on_n_faces++;
+
+	  double scal2 = v2n * faces[i].n;
+	  res = DOES_INTERSECT;
+	  if (scal2 > eps) res = IS_OUTSIDE;
+	  if (scal2 < -eps) res = IS_INSIDE;
+	}
+    }
+
+  if (point_on_n_faces == 1)
+    return res;
+
+
+
+
+  return Primitive :: VecInSolid2 (p, v1, v2, eps);
 }
 
+
+
+
 void Polyhedra :: GetPrimitiveData (char *& classname, 
 				    ARRAY<double> & coeffs) const
 {
@@ -182,19 +263,10 @@ void Polyhedra :: SetPrimitiveData (ARRAY<double> & coeffs)
 
 void Polyhedra :: Reduce (const BoxSphere<3> & box)
 {
-  int i;
-
-  for (i = 0; i < planes.Size(); i++)
+  for (int i = 0; i < planes.Size(); i++)
     surfaceactive[i] = 0;
 
-  /*
-  for (i = 1; i <= faces.Size(); i++)
-    if (FaceBoxIntersection (i, box))
-      surfaceactive.Elem (faces.Get(i).planenr) = 1;
-  */
-
-  for (i = 0; i < faces.Size(); i++)
-    //    if (faces.Get(i).bbox.Intersect (box))
+  for (int i = 0; i < faces.Size(); i++)
     if (FaceBoxIntersection (i, box))
       surfaceactive[faces[i].planenr] = 1;
 }
@@ -212,8 +284,6 @@ int Polyhedra :: AddPoint (const Point<3> & p)
 
 int Polyhedra :: AddFace (int pi1, int pi2, int pi3)
 {
-  int i;
-
   faces.Append (Face (pi1, pi2, pi3, points));
   
   Point<3> p1 = points[pi1];
@@ -229,7 +299,7 @@ int Polyhedra :: AddFace (int pi1, int pi2, int pi3)
   Plane pl (p1, n);
   int inverse;
   int identicto = -1;
-  for (i = 0; i < planes.Size(); i++)
+  for (int i = 0; i < planes.Size(); i++)
     if (pl.IsIdentic (*planes[i], inverse, 1e-6))
       {
 	if (!inverse)
diff --git a/Netgen/libsrc/csg/polyhedra.hpp b/Netgen/libsrc/csg/polyhedra.hpp
index d422c8ca15..529cfff40d 100644
--- a/Netgen/libsrc/csg/polyhedra.hpp
+++ b/Netgen/libsrc/csg/polyhedra.hpp
@@ -48,6 +48,12 @@ public:
 				   const Vec<3> & v,
 				   double eps) const;
 
+  // checks if lim s->0 lim t->0  p + t(v1 + s v2) in solid
+  virtual INSOLID_TYPE VecInSolid2 (const Point<3> & p,
+				    const Vec<3> & v1,
+				    const Vec<3> & v2,
+				    double eps) const;
+
   virtual int GetNSurfaces() const 
     { return planes.Size(); }
   virtual Surface & GetSurface (int i) 
diff --git a/Netgen/libsrc/csg/singularref.cpp b/Netgen/libsrc/csg/singularref.cpp
index b445d0085b..add7bfdf65 100644
--- a/Netgen/libsrc/csg/singularref.cpp
+++ b/Netgen/libsrc/csg/singularref.cpp
@@ -32,23 +32,21 @@ SingularEdge :: SingularEdge (double abeta,
 void SingularEdge :: FindPointsOnEdge (class Mesh & mesh)
 {
   (*testout) << "find points on edge" << endl;
-  int i, j;
+  int j;
   points.SetSize(0);
   segms.SetSize(0);
-  for (i = 1; i <= mesh.GetNSeg(); i++)
+  for (SegmentIndex si = 0; si < mesh.GetNSeg(); si++)
     {
-      INDEX_2 i2 (mesh.LineSegment(i).p1,
-		  mesh.LineSegment(i).p2);
+      INDEX_2 i2 (mesh[si].p1, mesh[si].p2);
       
-      int onedge = 1;
+      bool onedge = 1;
       for (j = 1; j <= 2; j++)
 	{
-	  const Point<3> p = mesh.Point(i2.I(j));
+	  const Point<3> p = mesh[ PointIndex (i2.I(j)) ];
 	  if (sol1->IsIn (p, 1e-3) && sol2->IsIn(p, 1e-3) &&
 	      !sol1->IsStrictIn (p, 1e-3) && !sol2->IsStrictIn(p, 1e-3))
 	    {
 	      ;
-
 	    }
 	  else
 	    onedge = 0;
@@ -56,44 +54,30 @@ void SingularEdge :: FindPointsOnEdge (class Mesh & mesh)
       if (onedge)
 	{
 	  segms.Append (i2);
-	  cout << "sing segment << " << i2 << endl;
-	  points.Append (mesh.Point(i2.I1()));
-	  points.Append (mesh.Point(i2.I2()));
+	  PrintMessage (5, "sing segment ", i2.I1(), " - ", i2.I2());
+	  points.Append (mesh[ PointIndex (i2.I1())]);
+	  points.Append (mesh[ PointIndex (i2.I2())]);
+	  mesh[si].singedge_left = 1;
+	  mesh[si].singedge_right = 1;
 	}	    
     }
 
-  
+  /*  
   (*testout) << "Singular points:" << endl;
-  for (i = 1; i <= points.Size(); i++)
-    (*testout) << points.Get(i) << endl;
-  /*
-  for (i = 1; i <= mesh.GetNP(); i++)
-    {
-      const Point<3> p = mesh.Point(i);
-      if (sol1->IsIn (p) && sol2->IsIn(p) &&
-	  !sol1->IsStrictIn (p) && !sol2->IsStrictIn(p))
-	{
-	  points.Append (p);
-	  cout << "Point " << p << " is on singular edge" << endl;
-	}
-    }
+  for (int i = 0; i < points.Size(); i++)
+    (*testout) << points[i] << endl;
   */
 }
 
 void SingularEdge :: SetMeshSize (class Mesh & mesh, double globalh)
 {
-  int i;
-
   double hloc = pow (globalh, 1/beta);
-  for (i = 1; i <= points.Size(); i++)
-    mesh.RestrictLocalH (points.Get(i), hloc);
+  for (int i = 0; i < points.Size(); i++)
+    mesh.RestrictLocalH (points[i], hloc);
 }
 
 
 
-
-
-
 SingularPoint :: SingularPoint (double abeta, 
 				const Solid * asol1, 
 				const Solid * asol2,
@@ -108,16 +92,17 @@ SingularPoint :: SingularPoint (double abeta,
 
 void SingularPoint :: FindPoints (class Mesh & mesh)
 {
-  int i;
   points.SetSize(0);
-  for (i = 1; i <= mesh.GetNP(); i++)
+  for (PointIndex pi = PointIndex::BASE; 
+       pi < mesh.GetNP()+PointIndex::BASE; pi++)
     {
-      const Point<3> p = mesh.Point(i);
+      const Point<3> p = mesh[pi];
       if (sol1->IsIn (p) && sol2->IsIn(p) && sol3->IsIn(p) &&
 	  !sol1->IsStrictIn (p) && !sol2->IsStrictIn(p) && !sol3->IsStrictIn(p))
 	{
 	  points.Append (p);
-	  cout << "Point " << p << " is a singular point" << endl;
+	  PrintMessage (5, "Point (", p(0), ", ", p(1), ", ", p(2), ") is singular");
+	  mesh[pi].SetSingular();
 	}
     }  
 }
@@ -125,10 +110,8 @@ void SingularPoint :: FindPoints (class Mesh & mesh)
 
 void SingularPoint :: SetMeshSize (class Mesh & mesh, double globalh)
 {
-  int i;
-
   double hloc = pow (globalh, 1/beta);
-  for (i = 1; i <= points.Size(); i++)
+  for (int i = 1; i <= points.Size(); i++)
     mesh.RestrictLocalH (points.Get(i), hloc);  
 }
 }
diff --git a/Netgen/libsrc/csg/singularref.hpp b/Netgen/libsrc/csg/singularref.hpp
index b2d2f2d839..d52dcf8f8c 100644
--- a/Netgen/libsrc/csg/singularref.hpp
+++ b/Netgen/libsrc/csg/singularref.hpp
@@ -13,6 +13,28 @@
 
 
 
+/**
+   Singular Face.
+   Causes a bounday layer mesh refinement.
+   All elements in subdomain domnr will get a boundary layer
+   on faces sharing the solid sol
+ */
+class SingularFace 
+{
+public:
+  int domnr;
+  const Solid *sol;
+  // ARRAY<Point<3> > points;
+  // ARRAY<INDEX_2> segms;
+public:
+  SingularFace (int adomnr, const Solid * asol)
+    : domnr(adomnr), sol(asol) { ; }
+  const Solid * GetSolid() const { return sol; }
+  int GetDomainNr () const { return domnr; }
+};
+
+
+///
 class SingularEdge 
 {
 public:
@@ -27,6 +49,7 @@ public:
 };
 
 
+///
 class SingularPoint
 {
 public:
diff --git a/Netgen/libsrc/csg/solid.cpp b/Netgen/libsrc/csg/solid.cpp
index 5ea936c1d1..08d20f78a4 100644
--- a/Netgen/libsrc/csg/solid.cpp
+++ b/Netgen/libsrc/csg/solid.cpp
@@ -9,1277 +9,1209 @@ namespace netgen
   using namespace netgen;
 
 
-SolidIterator :: SolidIterator ()
-{
-  ;
-}
-
-SolidIterator :: ~SolidIterator ()
-{
-  ;
-}
-
-
-
-
-int Solid :: cntnames = 0;
-  
-Solid :: Solid (Primitive * aprim)
-{
-  op = TERM;
-  prim = aprim;
-  s1 = s2 = NULL;
-  maxh = 1e10;
-
   /*
-  cntnames ++;
-  name = new char[16];
-  sprintf (name, "noname%d", cntnames);
+  SolidIterator :: SolidIterator ()
+  {
+    ;
+  }
+
+  SolidIterator :: ~SolidIterator ()
+  {
+    ;
+  }
   */
-  name = NULL;   // name by number ?
-}
 
-Solid :: Solid (optyp aop, Solid * as1, Solid * as2)
-{
-  op = aop;
-  s1 = as1;
-  s2 = as2;
-  prim = NULL;
-  name = NULL;
-  maxh = 1e10;
-}
 
-Solid :: ~Solid ()
-{
-  delete [] name;
 
-  switch (op)
-    {
-    case UNION:
-    case SECTION:
-      {
-	delete s1;
-	delete s2;
+  // int Solid :: cntnames = 0;
+  
+  Solid :: Solid (Primitive * aprim)
+  {
+    op = TERM;
+    prim = aprim;
+    s1 = s2 = NULL;
+    maxh = 1e10;
+    name = NULL;   
+  }
+
+  Solid :: Solid (optyp aop, Solid * as1, Solid * as2)
+  {
+    op = aop;
+    s1 = as1;
+    s2 = as2;
+    prim = NULL;
+    name = NULL;
+    maxh = 1e10;
+  }
+
+  Solid :: ~Solid ()
+  {
+    // cout << "delete solid, op = " << int(op) << endl;
+    delete [] name;
+
+    switch (op)
+      {
+      case UNION:
+      case SECTION:
+	{
+	  if (s1->op != ROOT) delete s1;
+	  if (s2->op != ROOT) delete s2;
+	  break;
+	}
+      case SUB:
+	// case ROOT:
+	{
+	  if (s1->op != ROOT) delete s1;
+	  break;
+	}
+      case TERM:
+	{
+	  // cout << "has term" << endl;
+	  delete prim;
+	  break;
+	}
+      default:
 	break;
       }
-    case SUB:
-      //    case ROOT:
+  }
+
+  void Solid :: SetName (const char * aname)
+  {
+    delete [] name;
+    name = new char[strlen (aname)+1];
+    strcpy (name, aname);
+  }
+
+
+  Solid * Solid :: Copy (CSGeometry & geom) const
+  {
+    Solid * nsol;
+    switch (op)
       {
-	delete s1;
-	break;
+      case TERM: case TERM_REF:
+	{
+	  Primitive * nprim = prim->Copy();
+	  geom.AddSurfaces (nprim);
+	  nsol = new Solid (nprim);
+	  break;
+	}
+
+      case SECTION:
+      case UNION:
+	{
+	  nsol = new Solid (op, s1->Copy(geom), s2->Copy(geom));
+	  break;
+	}
+
+      case SUB:
+	{
+	  nsol = new Solid (SUB, s1 -> Copy (geom));
+	  break;
+	}
+      
+      case ROOT:
+	{
+	  nsol = s1->Copy(geom);
+	  break;
+	}
       }
-    default:
-      break;
-    }
-}
 
-void Solid :: SetName (const char * aname)
-{
-  delete [] name;
-  name = new char[strlen (aname)+1];
-  strcpy (name, aname);
-}
+    return nsol;
+  }
+
+ 
+  void Solid :: Transform (Transformation<3> & trans)
+  {
+    switch (op)
+      {
+      case TERM: case TERM_REF:
+	{
+	  prim -> Transform (trans);
+	  break;
+	}
+      case SECTION:
+      case UNION:
+	{
+	  s1 -> Transform (trans);
+	  s2 -> Transform (trans);
+	  break;
+	}
 
+      case SUB:
+      case ROOT:
+	{
+	  s1 -> Transform (trans);
+	  break;
+	}
+      }  
+  }
 
 
 
-Solid * Solid :: Copy (CSGeometry & geom) const
-{
-  Solid * nsol;
-  switch (op)
-    {
-    case TERM:
+  void Solid :: IterateSolid (SolidIterator & it,
+			      bool only_once)
+  {
+    if (only_once)
       {
-	Primitive * nprim = prim->Copy();
+	if (visited)
+	  return;
 
-	for (int j = 0; j < nprim->GetNSurfaces(); j++)
-	  {
-	    geom.AddSurface (&nprim->GetSurface(j));
-	    nprim->SetSurfaceId (j, geom.GetNSurf()-1);
-	  }
-	nsol = new Solid (nprim);
-	break;
+	visited = 1; 
       }
-    case SECTION:
-    case UNION:
+
+    it.Do (this);
+
+    switch (op)
       {
-	Solid *nsol1, *nsol2;
+      case SECTION:
+	{
+	  s1->IterateSolid (it, only_once);
+	  s2->IterateSolid (it, only_once);
+	  break;
+	}
+      case UNION:
+	{
+	  s1->IterateSolid (it, only_once);
+	  s2->IterateSolid (it, only_once);
+	  break;
+	}
+      case SUB:
+      case ROOT:
+	{
+	  s1->IterateSolid (it, only_once);
+	  break;
+	}
+      } 
+  }
 
-	nsol1 = s1 -> Copy (geom);
-	nsol2 = s2 -> Copy (geom);
 
-	nsol = new Solid (op, nsol1, nsol2);
-	break;
-      }
 
-    case SUB:
+
+  bool Solid :: IsIn (const Point<3> & p, double eps) const
+  {
+    switch (op)
       {
-	Solid * nsol1 = s1 -> Copy (geom);
-	nsol = new Solid (SUB, nsol1);
-	break;
+      case TERM: case TERM_REF:
+	{
+	  INSOLID_TYPE ist = prim->PointInSolid (p, eps);
+	  return ( (ist == IS_INSIDE) || (ist == DOES_INTERSECT) ) ? 1 : 0;
+	}
+      case SECTION:
+	return s1->IsIn (p, eps) && s2->IsIn (p, eps);
+      case UNION:
+	return s1->IsIn (p, eps) || s2->IsIn (p, eps);
+      case SUB:
+	return !s1->IsStrictIn (p, eps);
+      case ROOT:
+	return s1->IsIn (p, eps);
       }
-      
-    case ROOT:
+    return 0;
+  }
+
+  bool Solid :: IsStrictIn (const Point<3> & p, double eps) const
+  {
+    switch (op)
       {
-	nsol = s1->Copy(geom);
-	break;
+      case TERM: case TERM_REF:
+	{
+	  INSOLID_TYPE ist = prim->PointInSolid (p, eps);
+	  return (ist == IS_INSIDE) ? 1 : 0;
+	}
+      case SECTION:
+	return s1->IsStrictIn(p, eps) && s2->IsStrictIn(p, eps);
+      case UNION:
+	return s1->IsStrictIn(p, eps) || s2->IsStrictIn(p, eps);
+      case SUB:
+	return !s1->IsIn (p, eps);
+      case ROOT:
+	return s1->IsStrictIn (p, eps);
       }
-    }
-  return nsol;
-}
+    return 0;
+  }
 
- 
-void Solid :: Transform (Transformation<3> & trans)
-{
-  switch (op)
-    {
-    case TERM:
+  bool Solid :: VectorIn (const Point<3> & p, const Vec<3> & v, 
+			 double eps) const
+  {
+    Vec<3> hv;
+    switch (op)
       {
-	prim -> Transform (trans);
-	break;
+      case TERM: case TERM_REF:
+	{
+	  INSOLID_TYPE ist = prim->VecInSolid (p, v, eps);
+	  return (ist == IS_INSIDE || ist == DOES_INTERSECT) ? 1 : 0;
+	}
+      case SECTION:
+	return s1 -> VectorIn (p, v, eps) && s2 -> VectorIn (p, v, eps);
+      case UNION:
+	return s1 -> VectorIn (p, v, eps) || s2 -> VectorIn (p, v, eps);
+      case SUB:
+	return !s1->VectorStrictIn(p, v, eps);
+      case ROOT:
+	return s1->VectorIn(p, v, eps);
       }
-    case SECTION:
-    case UNION:
+    return 0;
+  }
+
+  bool Solid :: VectorStrictIn (const Point<3> & p, const Vec<3> & v,
+			       double eps) const
+  {
+    Vec<3> hv;
+    switch (op)
       {
-	s1 -> Transform (trans);
-	s2 -> Transform (trans);
-	break;
+      case TERM: case TERM_REF:
+	{
+	  INSOLID_TYPE ist = prim->VecInSolid (p, v, eps);
+	  return (ist == IS_INSIDE) ? 1 : 0;
+	}
+      case SECTION:
+	return s1 -> VectorStrictIn (p, v, eps) && 
+	  s2 -> VectorStrictIn (p, v, eps);
+      case UNION:
+	return s1 -> VectorStrictIn (p, v, eps) || 
+	  s2 -> VectorStrictIn (p, v, eps);
+      case SUB:
+	return !s1->VectorIn(p, v, eps);
+      case ROOT:
+	return s1->VectorStrictIn(p, v, eps);
       }
+    return 0;
+  }
+
+
+  bool Solid::VectorIn2 (const Point<3> & p, const Vec<3> & v1, 
+			const Vec<3> & v2, double eps) const
+  {
+    if (VectorStrictIn (p, v1, eps))
+      return 1;
+    if (!VectorIn (p, v1, eps))
+      return 0;
+  
+    bool res = VectorIn2Rec (p, v1, v2, eps);
+    return res;
+  }
 
-    case SUB:
-    case ROOT:
+  bool Solid::VectorIn2Rec (const Point<3> & p, const Vec<3> & v1, 
+			   const Vec<3> & v2, double eps) const
+  {
+    switch (op)
       {
-	s1 -> Transform (trans);
-	break;
+      case TERM: case TERM_REF:
+	return prim->VecInSolid2 (p, v1, v2, eps);
+      case SECTION:
+	return s1->VectorIn2Rec (p, v1, v2, eps) && 
+	  s2->VectorIn2Rec (p, v1, v2, eps);
+      case UNION:
+	return s1->VectorIn2Rec (p, v1, v2, eps) ||
+	  s2->VectorIn2Rec (p, v1, v2, eps);
+      case SUB:
+	return !s1->VectorIn2Rec (p, v1, v2, eps);
+      case ROOT:
+	return s1->VectorIn2Rec (p, v1, v2, eps);
       }
-    }  
-}
+    return 0;  
+  }
 
 
 
-void Solid :: IterateSolid (SolidIterator & it,
-			    int only_once)
-{
-  if (only_once)
-    {
-      if (visited)
-	return;
 
-      visited = 1; 
-    }
 
-  it.Do (this);
 
-  switch (op)
-    {
-    case SECTION:
-      {
-	s1->IterateSolid (it, only_once);
-	s2->IterateSolid (it, only_once);
-	break;
-      }
-    case UNION:
+  void Solid :: Print (ostream & str) const
+  {
+    switch (op)
       {
-	s1->IterateSolid (it, only_once);
-	s2->IterateSolid (it, only_once);
-	break;
+      case TERM: case TERM_REF:
+	{
+	  str << prim->GetSurfaceId(0);
+	  for (int i = 1; i < prim->GetNSurfaces(); i++)
+	    str << "," << prim->GetSurfaceId(i);
+	  break;
+	}
+      case SECTION:
+	{
+	  str << "(";
+	  s1 -> Print (str);
+	  str << " AND ";
+	  s2 -> Print (str);
+	  str << ")";
+	  break;
+	}
+      case UNION:
+	{
+	  str << "(";
+	  s1 -> Print (str);
+	  str << " OR ";
+	  s2 -> Print (str);
+	  str << ")";
+	  break;
+	}
+      case SUB:
+	{
+	  str << " NOT ";
+	  s1 -> Print (str);
+	  break;
+	}
+      case ROOT:
+	{
+	  str << " [" << name << "=";
+	  s1 -> Print (str);
+	  str << "] ";
+	  break;
+	}
       }
-    case SUB:
-    case ROOT:
+  }
+
+
+
+  void Solid :: GetSolidData (ostream & ost, int first) const
+  {
+    switch (op)
       {
-	s1->IterateSolid (it, only_once);
-	break;
+      case SECTION:
+	{
+	  ost << "(";
+	  s1 -> GetSolidData (ost, 0);
+	  ost << " AND ";
+	  s2 -> GetSolidData (ost, 0);
+	  ost << ")";
+	  break;
+	}
+      case UNION:
+	{
+	  ost << "(";
+	  s1 -> GetSolidData (ost, 0);
+	  ost << " OR ";
+	  s2 -> GetSolidData (ost, 0);
+	  ost << ")";
+	  break;
+	}
+      case SUB:
+	{
+	  ost << "NOT ";
+	  s1 -> GetSolidData (ost, 0);
+	  break;
+	}
+      case TERM: case TERM_REF:
+	{
+	  if (name)
+	    ost << name;
+	  else
+	    ost << "(noname)";
+	  break;
+	}
+      case ROOT:
+	{
+	  if (first)
+	    s1 -> GetSolidData (ost, 0);
+	  else
+	    ost << name;
+	  break;
+	}
       }
-    } 
-}
+  }
 
 
 
+  static Solid * CreateSolidExpr (istream & ist, const SYMBOLTABLE<Solid*> & solids);
+  static Solid * CreateSolidTerm (istream & ist, const SYMBOLTABLE<Solid*> & solids);
+  static Solid * CreateSolidPrim (istream & ist, const SYMBOLTABLE<Solid*> & solids);
 
-int Solid :: IsIn (const Point<3> & p, double eps) const
-{
-  int hv1, hv2;
-  switch (op)
-    {
-    case TERM:
-      {
-	INSOLID_TYPE ist = prim->PointInSolid (p, eps);
-	return ( (ist == IS_INSIDE) || (ist == DOES_INTERSECT) ) ? 1 : 0;
-      }
-    case SECTION:
-      return s1->IsIn (p, eps) && s2->IsIn (p, eps);
-    case UNION:
-      return s1->IsIn (p, eps) || s2->IsIn (p, eps);
-    case SUB:
-      return !s1->IsStrictIn (p, eps);
-    case ROOT:
-      return s1->IsIn (p, eps);
-    }
-  return 0;
-}
+  static void ReadString (istream & ist, char * str)
+  {
+    char * hstr = str;
+    char ch;
 
-int Solid :: IsStrictIn (const Point<3> & p, double eps) const
-{
-  switch (op)
-    {
-    case TERM:
+    while (1)
       {
-	INSOLID_TYPE ist = prim->PointInSolid (p, eps);
-	return (ist == IS_INSIDE) ? 1 : 0;
-      }
-    case SECTION:
-      return s1->IsStrictIn(p, eps) && s2->IsStrictIn(p, eps);
-    case UNION:
-      return s1->IsStrictIn(p, eps) || s2->IsStrictIn(p, eps);
-    case SUB:
-      return !s1->IsIn (p, eps);
-    case ROOT:
-      return s1->IsStrictIn (p, eps);
-    }
-  return 0;
-}
+	ist.get(ch);
+	if (!ist.good()) break;
 
-int Solid :: VectorIn (const Point<3> & p, const Vec<3> & v, 
-		       double eps) const
-{
-  Vec<3> hv;
-  switch (op)
-    {
-    case TERM:
-      {
-	INSOLID_TYPE ist = prim->VecInSolid (p, v, eps);
-	return (ist == IS_INSIDE || ist == DOES_INTERSECT) ? 1 : 0;
+	if (!isspace (ch))
+	  {
+	    ist.putback (ch);
+	    break;
+	  }
       }
-    case SECTION:
-      return s1 -> VectorIn (p, v, eps) && s2 -> VectorIn (p, v, eps);
-    case UNION:
-      return s1 -> VectorIn (p, v, eps) || s2 -> VectorIn (p, v, eps);
-    case SUB:
-      return !s1->VectorStrictIn(p, v, eps);
-    case ROOT:
-      return s1->VectorIn(p, v, eps);
-    }
-  return 0;
-}
 
-int Solid :: VectorStrictIn (const Point<3> & p, const Vec<3> & v,
-			     double eps) const
-{
-  Vec<3> hv;
-  switch (op)
-    {
-    case TERM:
+    while (1)
       {
-	INSOLID_TYPE ist = prim->VecInSolid (p, v, eps);
-	return (ist == IS_INSIDE) ? 1 : 0;
+	ist.get(ch);
+	if (!ist.good()) break;
+	if (isalpha(ch) || isdigit(ch))
+	  {
+	    *str = ch;
+	    str++;
+	  }
+	else
+	  {
+	    ist.putback (ch);
+	    break;
+	  }
       }
-    case SECTION:
-      return s1 -> VectorStrictIn (p, v, eps) && 
-	s2 -> VectorStrictIn (p, v, eps);
-    case UNION:
-      return s1 -> VectorStrictIn (p, v, eps) || 
-	s2 -> VectorStrictIn (p, v, eps);
-    case SUB:
-      return !s1->VectorIn(p, v, eps);
-    case ROOT:
-      return s1->VectorStrictIn(p, v, eps);
-    }
-  return 0;
-}
+    *str = 0;
+    //  cout << "Read string (" << hstr << ")" 
+    //       << "put back: " << ch << endl;
+  }
 
 
-int Solid::VectorIn2 (const Point<3> & p, const Vec<3> & v1, 
-		      const Vec<3> & v2, double eps) const
-{
-  if (VectorStrictIn (p, v1, eps))
-    return 1;
-  if (!VectorIn (p, v1, eps))
-    return 0;
-  
-  return VectorIn2Rec (p, v1, v2, eps);
-}
+  Solid * CreateSolidExpr (istream & ist, const SYMBOLTABLE<Solid*> & solids)
+  {
+    //  cout << "create expr" << endl;
 
-int Solid::VectorIn2Rec (const Point<3> & p, const Vec<3> & v1, 
-			 const Vec<3> & v2, double eps) const
-{
-  int hv1, hv2;
-  switch (op)
-    {
-    case TERM:
-      return prim->VecInSolid2 (p, v1, v2, eps);
-    case SECTION:
-      return s1->VectorIn2Rec (p, v1, v2, eps) && 
-	s2->VectorIn2Rec (p, v1, v2, eps);
-    case UNION:
-      return s1->VectorIn2Rec (p, v1, v2, eps) ||
-	s2->VectorIn2Rec (p, v1, v2, eps);
-    case SUB:
-      return !s1->VectorIn2Rec (p, v1, v2, eps);
-    case ROOT:
-      return s1->VectorIn2Rec (p, v1, v2, eps);
-    }
-  return 0;  
-}
+    Solid *s1, *s2;
+    char str[100];
 
+    s1 = CreateSolidTerm (ist, solids);
+    ReadString (ist, str);
+    if (strcmp (str, "OR") == 0)
+      {
+	//      cout << " OR ";
+	s2 = CreateSolidExpr (ist, solids);
+	return new Solid (Solid::UNION, s1, s2);
+      }
 
+    //  cout << "no OR found, put back string: " << str << endl;
+    int i;
+    for (i = strlen(str)-1; i >= 0; i--)
+      ist.putback (str[i]);
 
+    return s1;
+  }
 
+  Solid * CreateSolidTerm (istream & ist, const SYMBOLTABLE<Solid*> & solids)
+  {
+    //  cout << "create term" << endl;
 
+    Solid *s1, *s2;
+    char str[100];
 
-void Solid :: Print (ostream & str) const
-{
-  switch (op)
-    {
-    case TERM:
-      {
-	str << prim->GetSurfaceId(0);
-	for (int i = 1; i < prim->GetNSurfaces(); i++)
-	  str << "," << prim->GetSurfaceId(i);
-	break;
-      }
-    case SECTION:
-      {
-	str << "(";
-	s1 -> Print (str);
-	str << " AND ";
-	s2 -> Print (str);
-	str << ")";
-	break;
-      }
-    case UNION:
-      {
-	str << "(";
-	s1 -> Print (str);
-	str << " OR ";
-	s2 -> Print (str);
-	str << ")";
-	break;
-      }
-    case SUB:
-      {
-	str << " NOT ";
-	s1 -> Print (str);
-	break;
-      }
-    case ROOT:
+    s1 = CreateSolidPrim (ist, solids);
+    ReadString (ist, str);
+    if (strcmp (str, "AND") == 0)
       {
-	str << " [" << name << "=";
-	s1 -> Print (str);
-	str << "] ";
-	break;
+	//      cout << " AND ";
+	s2 = CreateSolidTerm (ist, solids);
+	return new Solid (Solid::SECTION, s1, s2);
       }
-    }
-}
 
 
+    //  cout << "no AND found, put back string: " << str << endl;
+    int i;
+    for (i = strlen(str)-1; i >= 0; i--)
+      ist.putback (str[i]);
 
-void Solid :: GetSolidData (ostream & ost, int first) const
-{
-  switch (op)
-    {
-    case SECTION:
-      {
-	ost << "(";
-	s1 -> GetSolidData (ost, 0);
-	ost << " AND ";
-	s2 -> GetSolidData (ost, 0);
-	ost << ")";
-	break;
-      }
-    case UNION:
-      {
-	ost << "(";
-	s1 -> GetSolidData (ost, 0);
-	ost << " OR ";
-	s2 -> GetSolidData (ost, 0);
-	ost << ")";
-	break;
-      }
-    case SUB:
+    return s1;
+  }
+
+  Solid * CreateSolidPrim (istream & ist, const SYMBOLTABLE<Solid*> & solids)
+  {
+    Solid * s1;
+    char ch;
+    char str[100];
+
+    ist >> ch;
+    if (ch == '(')
       {
-	ost << "NOT ";
-	s1 -> GetSolidData (ost, 0);
-	break;
+	s1 = CreateSolidExpr (ist, solids);
+	ist >> ch;  // ')'
+	//      cout << "close back " << ch << endl;
+	return s1;
       }
-    case TERM:
+    ist.putback (ch);
+  
+    ReadString (ist, str);
+    if (strcmp (str, "NOT") == 0)
       {
-	if (name)
-	  ost << name;
-	else
-	  ost << "(noname)";
-	break;
+	//      cout << " NOT ";
+	s1 = CreateSolidPrim (ist, solids);
+	return new Solid (Solid::SUB, s1);
       }
-    case ROOT:
+
+    (*testout) << "get terminal " << str << endl;
+    s1 = solids.Get(str);
+    if (s1)
       {
-	if (first)
-	  s1 -> GetSolidData (ost, 0);
-	else
-	  ost << name;
-	break;
+	//      cout << "primitive: " << str << endl;
+	return s1;
       }
-    }
-}
+    cerr << "syntax error" << endl;
 
+    return NULL;
+  }
 
 
-static Solid * CreateSolidExpr (istream & ist, const SYMBOLTABLE<Solid*> & solids);
-static Solid * CreateSolidTerm (istream & ist, const SYMBOLTABLE<Solid*> & solids);
-static Solid * CreateSolidPrim (istream & ist, const SYMBOLTABLE<Solid*> & solids);
+  Solid * Solid :: CreateSolid (istream & ist, const SYMBOLTABLE<Solid*> & solids)
+  {
+    Solid * nsol =  CreateSolidExpr (ist, solids);
+    nsol = new Solid (ROOT, nsol);
+    (*testout) << "Print new sol: ";
+    nsol -> Print (*testout);
+    (*testout) << endl;
+    return nsol;
+  }
 
-static void ReadString (istream & ist, char * str)
-{
-  char * hstr = str;
-  char ch;
 
-  while (1)
-    {
-      ist.get(ch);
-      if (!ist.good()) break;
 
-      if (!isspace (ch))
+  void Solid :: Boundaries (const Point<3> & p, ARRAY<int> & bounds) const
+  {
+    int in, strin;
+    bounds.SetSize (0);
+    RecBoundaries (p, bounds, in, strin);
+  }
+
+  void Solid :: RecBoundaries (const Point<3> & p, ARRAY<int> & bounds,
+			       int & in, int & strin) const
+  {
+    switch (op)
+      {
+      case TERM: case TERM_REF:
 	{
-	  ist.putback (ch);
+	  /*
+	    double val;
+	    val = surf->CalcFunctionValue (p);
+	    in = (val < 1e-6);
+	    strin = (val < -1e-6);
+	    if (in && !strin) bounds.Append (id);
+	  */
+	  if (prim->PointInSolid (p, 1e-6) == DOES_INTERSECT)
+	    bounds.Append (prim->GetSurfaceId (1));
 	  break;
 	}
-    }
+      case SECTION:
+	{
+	  int i, in1, in2, strin1, strin2;
+	  ARRAY<int> bounds1, bounds2;
+
+	  s1 -> RecBoundaries (p, bounds1, in1, strin1);
+	  s2 -> RecBoundaries (p, bounds2, in2, strin2);
 
-  while (1)
-    {
-      ist.get(ch);
-      if (!ist.good()) break;
-      if (isalpha(ch) || isdigit(ch))
+	  if (in1 && in2)
+	    {
+	      for (i = 1; i <= bounds1.Size(); i++)
+		bounds.Append (bounds1.Get(i));
+	      for (i = 1; i <= bounds2.Size(); i++)
+		bounds.Append (bounds2.Get(i));
+	    }
+	  in = (in1 && in2);
+	  strin = (strin1 && strin2);
+	  break;
+	}
+      case UNION:
 	{
-	  *str = ch;
-	  str++;
+	  int i, in1, in2, strin1, strin2;
+	  ARRAY<int> bounds1, bounds2;
+
+	  s1 -> RecBoundaries (p, bounds1, in1, strin1);
+	  s2 -> RecBoundaries (p, bounds2, in2, strin2);
+
+	  if (!strin1 && !strin2)
+	    {
+	      for (i = 1; i <= bounds1.Size(); i++)
+		bounds.Append (bounds1.Get(i));
+	      for (i = 1; i <= bounds2.Size(); i++)
+		bounds.Append (bounds2.Get(i));
+	    }
+	  in = (in1 || in2);
+	  strin = (strin1 || strin2);
+	  break;
 	}
-      else
+      case SUB:
 	{
-	  ist.putback (ch);
+	  int hin, hstrin;
+	  s1 -> RecBoundaries (p, bounds, hin, hstrin);
+	  in = !hstrin;
+	  strin = !hin;
 	  break;
 	}
-    }
-  *str = 0;
-  //  cout << "Read string (" << hstr << ")" 
-  //       << "put back: " << ch << endl;
-}
 
+      case ROOT:
+	{
+	  s1 -> RecBoundaries (p, bounds, in, strin);
+	  break;
+	}
+      }
+  }
 
-Solid * CreateSolidExpr (istream & ist, const SYMBOLTABLE<Solid*> & solids)
-{
-  //  cout << "create expr" << endl;
-
-  Solid *s1, *s2;
-  char str[100];
-
-  s1 = CreateSolidTerm (ist, solids);
-  ReadString (ist, str);
-  if (strcmp (str, "OR") == 0)
-    {
-      //      cout << " OR ";
-      s2 = CreateSolidExpr (ist, solids);
-      return new Solid (Solid::UNION, s1, s2);
-    }
-
-  //  cout << "no OR found, put back string: " << str << endl;
-  int i;
-  for (i = strlen(str)-1; i >= 0; i--)
-    ist.putback (str[i]);
-
-  return s1;
-}
 
-Solid * CreateSolidTerm (istream & ist, const SYMBOLTABLE<Solid*> & solids)
-{
-  //  cout << "create term" << endl;
 
-  Solid *s1, *s2;
-  char str[100];
+  void Solid :: TangentialSolid (const Point<3> & p, Solid *& tansol) const
+  {
+    int in, strin;
+    RecTangentialSolid (p, tansol, in, strin);
+  }
 
-  s1 = CreateSolidPrim (ist, solids);
-  ReadString (ist, str);
-  if (strcmp (str, "AND") == 0)
-    {
-      //      cout << " AND ";
-      s2 = CreateSolidTerm (ist, solids);
-      return new Solid (Solid::SECTION, s1, s2);
-    }
+  void Solid :: RecTangentialSolid (const Point<3> & p, Solid *& tansol,
+				    int & in, int & strin) const
+  {
+    tansol = NULL;
 
+    switch (op)
+      {
+      case TERM: case TERM_REF:
+	{
+	  /*
+	    double val;
+	    val = surf->CalcFunctionValue (p);
+	    in = (val < 1e-6);
+	    strin = (val < -1e-6);
+	    if (in && !strin)
+	    tansol = new Solid (surf, id);
+	  */
 
-  //  cout << "no AND found, put back string: " << str << endl;
-  int i;
-  for (i = strlen(str)-1; i >= 0; i--)
-    ist.putback (str[i]);
+	  INSOLID_TYPE ist = prim->PointInSolid(p, 1e-6);
 
-  return s1;
-}
+	  in = (ist == IS_INSIDE || ist == DOES_INTERSECT);
+	  strin = (ist == IS_INSIDE);
 
-Solid * CreateSolidPrim (istream & ist, const SYMBOLTABLE<Solid*> & solids)
-{
-  Solid * s1;
-  char ch;
-  char str[100];
-
-  ist >> ch;
-  if (ch == '(')
-    {
-      s1 = CreateSolidExpr (ist, solids);
-      ist >> ch;  // ')'
-      //      cout << "close back " << ch << endl;
-      return s1;
-    }
-  ist.putback (ch);
-  
-  ReadString (ist, str);
-  if (strcmp (str, "NOT") == 0)
-    {
-      //      cout << " NOT ";
-      s1 = CreateSolidPrim (ist, solids);
-      return new Solid (Solid::SUB, s1);
-    }
-
-  (*testout) << "get terminal " << str << endl;
-  s1 = solids.Get(str);
-  if (s1)
-    {
-      //      cout << "primitive: " << str << endl;
-      return s1;
-    }
-  cerr << "syntax error" << endl;
-
-  return NULL;
-}
+	  if (ist == DOES_INTERSECT)
+	    {
+	      tansol = new Solid (prim);
+	      tansol -> op = TERM_REF;
+	    }
+	  break;
+	}
+      case SECTION:
+	{
+	  int in1, in2, strin1, strin2;
+	  Solid * tansol1, * tansol2;
 
+	  s1 -> RecTangentialSolid (p, tansol1, in1, strin1);
+	  s2 -> RecTangentialSolid (p, tansol2, in2, strin2);
 
-Solid * Solid :: CreateSolid (istream & ist, const SYMBOLTABLE<Solid*> & solids)
-{
-  Solid * nsol =  CreateSolidExpr (ist, solids);
-  nsol = new Solid (ROOT, nsol);
-  (*testout) << "Print new sol: ";
-  nsol -> Print (*testout);
-  (*testout) << endl;
-  return nsol;
-}
+	  if (in1 && in2)
+	    {
+	      if (tansol1 && tansol2)
+		tansol = new Solid (SECTION, tansol1, tansol2);
+	      else if (tansol1)
+		tansol = tansol1;
+	      else if (tansol2)
+		tansol = tansol2;
+	    }
+	  in = (in1 && in2);
+	  strin = (strin1 && strin2);
+	  break;
+	}
+      case UNION:
+	{
+	  int in1, in2, strin1, strin2;
+	  Solid * tansol1, * tansol2;
 
+	  s1 -> RecTangentialSolid (p, tansol1, in1, strin1);
+	  s2 -> RecTangentialSolid (p, tansol2, in2, strin2);
 
+	  if (!strin1 && !strin2)
+	    {
+	      if (tansol1 && tansol2)
+		tansol = new Solid (UNION, tansol1, tansol2);
+	      else if (tansol1)
+		tansol = tansol1;
+	      else if (tansol2)
+		tansol = tansol2;
+	    }
+	  in = (in1 || in2);
+	  strin = (strin1 || strin2);
+	  break;
+	}
+      case SUB:
+	{
+	  int hin, hstrin;
+	  Solid * tansol1;
 
-void Solid :: Boundaries (const Point<3> & p, ARRAY<int> & bounds) const
-{
-  int in, strin;
-  bounds.SetSize (0);
-  RecBoundaries (p, bounds, in, strin);
-}
+	  s1 -> RecTangentialSolid (p, tansol1, hin, hstrin);
 
-void Solid :: RecBoundaries (const Point<3> & p, ARRAY<int> & bounds,
-			     int & in, int & strin) const
-{
-  switch (op)
-    {
-    case TERM:
-      {
-	/*
-	double val;
-	val = surf->CalcFunctionValue (p);
-	in = (val < 1e-6);
-	strin = (val < -1e-6);
-	if (in && !strin) bounds.Append (id);
-	*/
-	if (prim->PointInSolid (p, 1e-6) == DOES_INTERSECT)
-	  bounds.Append (prim->GetSurfaceId (1));
-	break;
-      }
-    case SECTION:
-      {
-	int i, in1, in2, strin1, strin2;
-	ARRAY<int> bounds1, bounds2;
-
-	s1 -> RecBoundaries (p, bounds1, in1, strin1);
-	s2 -> RecBoundaries (p, bounds2, in2, strin2);
-
-	if (in1 && in2)
-	  {
-	    for (i = 1; i <= bounds1.Size(); i++)
-	      bounds.Append (bounds1.Get(i));
-	    for (i = 1; i <= bounds2.Size(); i++)
-	      bounds.Append (bounds2.Get(i));
-	  }
-	in = (in1 && in2);
-	strin = (strin1 && strin2);
-	break;
-      }
-    case UNION:
-      {
-	int i, in1, in2, strin1, strin2;
-	ARRAY<int> bounds1, bounds2;
-
-	s1 -> RecBoundaries (p, bounds1, in1, strin1);
-	s2 -> RecBoundaries (p, bounds2, in2, strin2);
-
-	if (!strin1 && !strin2)
-	  {
-	    for (i = 1; i <= bounds1.Size(); i++)
-	      bounds.Append (bounds1.Get(i));
-	    for (i = 1; i <= bounds2.Size(); i++)
-	      bounds.Append (bounds2.Get(i));
-	  }
-	in = (in1 || in2);
-	strin = (strin1 || strin2);
-	break;
-      }
-    case SUB:
-      {
-	int hin, hstrin;
-	s1 -> RecBoundaries (p, bounds, hin, hstrin);
-	in = !hstrin;
-	strin = !hin;
-	break;
-      }
-
-    case ROOT:
-      {
-	s1 -> RecBoundaries (p, bounds, in, strin);
-	break;
-      }
-    }
-}
-
-
-
-void Solid :: TangentialSolid (const Point<3> & p, Solid *& tansol) const
-{
-  int in, strin;
-  RecTangentialSolid (p, tansol, in, strin);
-}
-
-void Solid :: RecTangentialSolid (const Point<3> & p, Solid *& tansol,
-				  int & in, int & strin) const
-{
-  tansol = NULL;
-
-  switch (op)
-    {
-    case TERM:
-      {
-	/*
-	double val;
-	val = surf->CalcFunctionValue (p);
-	in = (val < 1e-6);
-	strin = (val < -1e-6);
-	if (in && !strin)
-	  tansol = new Solid (surf, id);
-	*/
-
-	INSOLID_TYPE ist = prim->PointInSolid(p, 1e-6);
-
-	in = (ist == IS_INSIDE || ist == DOES_INTERSECT);
-	strin = (ist == IS_INSIDE);
-
-	if (ist == DOES_INTERSECT)
-	  tansol = new Solid (prim);
-	break;
-      }
-    case SECTION:
-      {
-	int in1, in2, strin1, strin2;
-	Solid * tansol1, * tansol2;
-
-	s1 -> RecTangentialSolid (p, tansol1, in1, strin1);
-	s2 -> RecTangentialSolid (p, tansol2, in2, strin2);
-
-	if (in1 && in2)
-	  {
-	    if (tansol1 && tansol2)
-	      tansol = new Solid (SECTION, tansol1, tansol2);
-	    else if (tansol1)
-	      tansol = tansol1;
-	    else if (tansol2)
-	      tansol = tansol2;
-	  }
-	in = (in1 && in2);
-	strin = (strin1 && strin2);
-	break;
-      }
-    case UNION:
-      {
-	int in1, in2, strin1, strin2;
-	Solid * tansol1, * tansol2;
-
-	s1 -> RecTangentialSolid (p, tansol1, in1, strin1);
-	s2 -> RecTangentialSolid (p, tansol2, in2, strin2);
-
-	if (!strin1 && !strin2)
-	  {
-	    if (tansol1 && tansol2)
-	      tansol = new Solid (UNION, tansol1, tansol2);
-	    else if (tansol1)
-	      tansol = tansol1;
-	    else if (tansol2)
-	      tansol = tansol2;
-	  }
-	in = (in1 || in2);
-	strin = (strin1 || strin2);
-	break;
+	  if (tansol1)
+	    tansol = new Solid (SUB, tansol1);
+	  in = !hstrin;
+	  strin = !hin;
+	  break;
+	}
+      case ROOT:
+	{
+	  s1 -> RecTangentialSolid (p, tansol, in, strin);
+	  break;
+	}
       }
-    case SUB:
-      {
-	int hin, hstrin;
-	Solid * tansol1;
+  }
 
-	s1 -> RecTangentialSolid (p, tansol1, hin, hstrin);
 
-	if (tansol1)
-	  tansol = new Solid (SUB, tansol1);
-	in = !hstrin;
-	strin = !hin;
-	break;
-      }
-    case ROOT:
-      {
-	s1 -> RecTangentialSolid (p, tansol, in, strin);
-	break;
-      }
-    }
-}
 
 
+  void Solid :: TangentialSolid2 (const Point<3> & p, 
+				  const Vec<3> & t,
+				  Solid *& tansol) const
+  {
+    int in, strin;
+    RecTangentialSolid2 (p, t, tansol, in, strin);
+  }
 
+  void Solid :: RecTangentialSolid2 (const Point<3> & p, const Vec<3> & t,
+				     Solid *& tansol,
+				     int & in, int & strin) const
+  {
+    tansol = NULL;
 
-void Solid :: TangentialSolid2 (const Point<3> & p, 
-				const Vec<3> & t,
-				Solid *& tansol) const
-{
-  int in, strin;
-  RecTangentialSolid2 (p, t, tansol, in, strin);
-}
-
-void Solid :: RecTangentialSolid2 (const Point<3> & p, const Vec<3> & t,
-				   Solid *& tansol,
-				   int & in, int & strin) const
-{
-  tansol = NULL;
-
-  switch (op)
-    {
-    case TERM:
-      {
-	/*
-	double val;
-	val = surf->CalcFunctionValue (p);
-	in = (val < 1e-6);
-	strin = (val < -1e-6);
-	if (in && !strin)
-	  tansol = new Solid (surf, id);
-	*/
-
-	INSOLID_TYPE ist = prim->PointInSolid(p, 1e-6);
-	if (ist == DOES_INTERSECT)
-	  ist = prim->VecInSolid (p, t, 1e-6);
-
-	in = (ist == IS_INSIDE || ist == DOES_INTERSECT);
-	strin = (ist == IS_INSIDE);
-
-	if (ist == DOES_INTERSECT)
-	  tansol = new Solid (prim);
-	break;
-      }
-    case SECTION:
+    switch (op)
       {
-	int in1, in2, strin1, strin2;
-	Solid * tansol1, * tansol2;
+      case TERM: case TERM_REF:
+	{
+	  /*
+	    double val;
+	    val = surf->CalcFunctionValue (p);
+	    in = (val < 1e-6);
+	    strin = (val < -1e-6);
+	    if (in && !strin)
+	    tansol = new Solid (surf, id);
+	  */
+
+	  INSOLID_TYPE ist = prim->PointInSolid(p, 1e-6);
+	  if (ist == DOES_INTERSECT)
+	    ist = prim->VecInSolid (p, t, 1e-6);
+
+	  in = (ist == IS_INSIDE || ist == DOES_INTERSECT);
+	  strin = (ist == IS_INSIDE);
+
+	  if (ist == DOES_INTERSECT)
+	    {
+	      tansol = new Solid (prim);
+	      tansol -> op = TERM_REF;
+	    }
+	  break;
+	}
+      case SECTION:
+	{
+	  int in1, in2, strin1, strin2;
+	  Solid * tansol1, * tansol2;
 
-	s1 -> RecTangentialSolid2 (p, t, tansol1, in1, strin1);
-	s2 -> RecTangentialSolid2 (p, t, tansol2, in2, strin2);
+	  s1 -> RecTangentialSolid2 (p, t, tansol1, in1, strin1);
+	  s2 -> RecTangentialSolid2 (p, t, tansol2, in2, strin2);
 
-	if (in1 && in2)
-	  {
-	    if (tansol1 && tansol2)
-	      tansol = new Solid (SECTION, tansol1, tansol2);
-	    else if (tansol1)
-	      tansol = tansol1;
-	    else if (tansol2)
-	      tansol = tansol2;
-	  }
-	in = (in1 && in2);
-	strin = (strin1 && strin2);
-	break;
-      }
-    case UNION:
-      {
-	int in1, in2, strin1, strin2;
-	Solid * tansol1, * tansol2;
+	  if (in1 && in2)
+	    {
+	      if (tansol1 && tansol2)
+		tansol = new Solid (SECTION, tansol1, tansol2);
+	      else if (tansol1)
+		tansol = tansol1;
+	      else if (tansol2)
+		tansol = tansol2;
+	    }
+	  in = (in1 && in2);
+	  strin = (strin1 && strin2);
+	  break;
+	}
+      case UNION:
+	{
+	  int in1, in2, strin1, strin2;
+	  Solid * tansol1, * tansol2;
 
-	s1 -> RecTangentialSolid2 (p, t, tansol1, in1, strin1);
-	s2 -> RecTangentialSolid2 (p, t, tansol2, in2, strin2);
+	  s1 -> RecTangentialSolid2 (p, t, tansol1, in1, strin1);
+	  s2 -> RecTangentialSolid2 (p, t, tansol2, in2, strin2);
 
-	if (!strin1 && !strin2)
-	  {
-	    if (tansol1 && tansol2)
-	      tansol = new Solid (UNION, tansol1, tansol2);
-	    else if (tansol1)
-	      tansol = tansol1;
-	    else if (tansol2)
-	      tansol = tansol2;
-	  }
-	in = (in1 || in2);
-	strin = (strin1 || strin2);
-	break;
-      }
-    case SUB:
-      {
-	int hin, hstrin;
-	Solid * tansol1;
+	  if (!strin1 && !strin2)
+	    {
+	      if (tansol1 && tansol2)
+		tansol = new Solid (UNION, tansol1, tansol2);
+	      else if (tansol1)
+		tansol = tansol1;
+	      else if (tansol2)
+		tansol = tansol2;
+	    }
+	  in = (in1 || in2);
+	  strin = (strin1 || strin2);
+	  break;
+	}
+      case SUB:
+	{
+	  int hin, hstrin;
+	  Solid * tansol1;
 
-	s1 -> RecTangentialSolid2 (p, t, tansol1, hin, hstrin);
+	  s1 -> RecTangentialSolid2 (p, t, tansol1, hin, hstrin);
 
-	if (tansol1)
-	  tansol = new Solid (SUB, tansol1);
-	in = !hstrin;
-	strin = !hin;
-	break;
-      }
-    case ROOT:
-      {
-	s1 -> RecTangentialSolid2 (p, t, tansol, in, strin);
-	break;
+	  if (tansol1)
+	    tansol = new Solid (SUB, tansol1);
+	  in = !hstrin;
+	  strin = !hin;
+	  break;
+	}
+      case ROOT:
+	{
+	  s1 -> RecTangentialSolid2 (p, t, tansol, in, strin);
+	  break;
+	}
       }
-    }
-}
+  }
 
 
 
 
-int Solid :: Edge (const Point<3> & p, const Vec<3> & v) const
-{
-  int in, strin, faces;
-  RecEdge (p, v, in, strin, faces);
-  return faces >= 2;
-}
+  int Solid :: Edge (const Point<3> & p, const Vec<3> & v) const
+  {
+    int in, strin, faces;
+    RecEdge (p, v, in, strin, faces);
+    return faces >= 2;
+  }
 
-int Solid :: OnFace (const Point<3> & p, const Vec<3> & v) const
-{
-  int in, strin, faces;
-  RecEdge (p, v, in, strin, faces);
-  return faces >= 1;
-}
+  int Solid :: OnFace (const Point<3> & p, const Vec<3> & v) const
+  {
+    int in, strin, faces;
+    RecEdge (p, v, in, strin, faces);
+    return faces >= 1;
+  }
 
 
-void Solid :: RecEdge (const Point<3> & p, const Vec<3> & v,
-                       int & in, int & strin, int & faces) const
-{
-  switch (op)
-    {
-    case TERM:
-      {
-	INSOLID_TYPE ist = prim->VecInSolid (p, v, 1e-6);
-	in = (ist == IS_INSIDE || ist == DOES_INTERSECT);
-	strin = (ist == IS_INSIDE);
-	/*
-	in = VectorIn (p, v);
-	strin = VectorStrictIn (p, v);
-	*/
-	faces = 0;
-
-	if (in && ! strin)
-	  {
-	    //	    faces = 1;
-	    int i; 
-	    Vec<3> grad;
-	    for (i = 1; i <= prim->GetNSurfaces(); i++)
-	      {
-		double val = prim->GetSurface(i).CalcFunctionValue(p);
-		prim->GetSurface(i).CalcGradient (p, grad);
-		if (fabs (val) < 1e-6 && fabs (v * grad) < 1e-6)
-		  faces++;
-	      }
-	  }
-	    //	else
-	    //	  faces = 0;
-	break;
-      }
-    case SECTION:
+  void Solid :: RecEdge (const Point<3> & p, const Vec<3> & v,
+			 int & in, int & strin, int & faces) const
+  {
+    switch (op)
       {
-	int in1, in2, strin1, strin2, faces1, faces2;
+      case TERM: case TERM_REF:
+	{
+	  INSOLID_TYPE ist = prim->VecInSolid (p, v, 1e-6);
+	  in = (ist == IS_INSIDE || ist == DOES_INTERSECT);
+	  strin = (ist == IS_INSIDE);
+	  /*
+	    in = VectorIn (p, v);
+	    strin = VectorStrictIn (p, v);
+	  */
+	  faces = 0;
+
+	  if (in && ! strin)
+	    {
+	      //	    faces = 1;
+	      int i; 
+	      Vec<3> grad;
+	      for (i = 0; i < prim->GetNSurfaces(); i++)
+		{
+		  double val = prim->GetSurface(i).CalcFunctionValue(p);
+		  prim->GetSurface(i).CalcGradient (p, grad);
+		  if (fabs (val) < 1e-6 && fabs (v * grad) < 1e-6)
+		    faces++;
+		}
+	    }
+	  //	else
+	  //	  faces = 0;
+	  break;
+	}
+      case SECTION:
+	{
+	  int in1, in2, strin1, strin2, faces1, faces2;
 
-	s1 -> RecEdge (p, v, in1, strin1, faces1);
-	s2 -> RecEdge (p, v, in2, strin2, faces2);
+	  s1 -> RecEdge (p, v, in1, strin1, faces1);
+	  s2 -> RecEdge (p, v, in2, strin2, faces2);
 
-	faces = 0;
-	if (in1 && in2)
-	  faces = faces1 + faces2;
-	in = in1 && in2;
-	strin = strin1 && strin2;
-	break;
-      }
-    case UNION:
-      {
-	int in1, in2, strin1, strin2, faces1, faces2;
+	  faces = 0;
+	  if (in1 && in2)
+	    faces = faces1 + faces2;
+	  in = in1 && in2;
+	  strin = strin1 && strin2;
+	  break;
+	}
+      case UNION:
+	{
+	  int in1, in2, strin1, strin2, faces1, faces2;
 
-	s1 -> RecEdge (p, v, in1, strin1, faces1);
-	s2 -> RecEdge (p, v, in2, strin2, faces2);
+	  s1 -> RecEdge (p, v, in1, strin1, faces1);
+	  s2 -> RecEdge (p, v, in2, strin2, faces2);
 
-	faces = 0;
-	if (!strin1 && !strin2)
-	  faces = faces1 + faces2;
-	in = in1 || in2;
-	strin = strin1 || strin2;
-	break;
-      }
-    case SUB:
-      {
-	int in1, strin1;
-	s1 -> RecEdge (p, v, in1, strin1, faces);
-	in = !strin1;
-	strin = !in1;
-	break;
-      }
-    case ROOT:
-      {
-	s1 -> RecEdge (p, v, in, strin, faces);
-	break;
+	  faces = 0;
+	  if (!strin1 && !strin2)
+	    faces = faces1 + faces2;
+	  in = in1 || in2;
+	  strin = strin1 || strin2;
+	  break;
+	}
+      case SUB:
+	{
+	  int in1, strin1;
+	  s1 -> RecEdge (p, v, in1, strin1, faces);
+	  in = !strin1;
+	  strin = !in1;
+	  break;
+	}
+      case ROOT:
+	{
+	  s1 -> RecEdge (p, v, in, strin, faces);
+	  break;
+	}
       }
-    }
-}
+  }
 
 
-void Solid :: CalcSurfaceInverse ()
-{
-  CalcSurfaceInverseRec (0);
-}
+  void Solid :: CalcSurfaceInverse ()
+  {
+    CalcSurfaceInverseRec (0);
+  }
 
-void Solid :: CalcSurfaceInverseRec (int inv)
-{
-  switch (op)
-    {
-    case TERM:
-      {
-	int priminv;
-	for (int i = 0; i < prim->GetNSurfaces(); i++)
-	  {
-	    priminv = prim->SurfaceInverted(i);
-	    if (inv) priminv = 1 - priminv;
-	    prim->GetSurface(i).SetInverse (priminv);
-	  }
-	break;
-      }
-    case UNION:
-    case SECTION:
-      {
-	s1 -> CalcSurfaceInverseRec (inv);
-	s2 -> CalcSurfaceInverseRec (inv);
-	break;
-      }
-    case SUB:
-      {
-	s1 -> CalcSurfaceInverseRec (1 - inv);
-	break;
-      }
-    case ROOT:
+  void Solid :: CalcSurfaceInverseRec (int inv)
+  {
+    switch (op)
       {
-	s1 -> CalcSurfaceInverseRec (inv);
-	break;
+      case TERM: case TERM_REF:
+	{
+	  int priminv;
+	  for (int i = 0; i < prim->GetNSurfaces(); i++)
+	    {
+	      priminv = prim->SurfaceInverted(i);
+	      if (inv) priminv = 1 - priminv;
+	      prim->GetSurface(i).SetInverse (priminv);
+	    }
+	  break;
+	}
+      case UNION:
+      case SECTION:
+	{
+	  s1 -> CalcSurfaceInverseRec (inv);
+	  s2 -> CalcSurfaceInverseRec (inv);
+	  break;
+	}
+      case SUB:
+	{
+	  s1 -> CalcSurfaceInverseRec (1 - inv);
+	  break;
+	}
+      case ROOT:
+	{
+	  s1 -> CalcSurfaceInverseRec (inv);
+	  break;
+	}
       }
-    }
-}
+  }
 
 
-Solid * Solid :: GetReducedSolid (const BoxSphere<3> & box) const
-{
-  int in;
-  Solid * res =  RecGetReducedSolid (box, in);
-  return res;
-}
+  Solid * Solid :: GetReducedSolid (const BoxSphere<3> & box) const
+  {
+    INSOLID_TYPE in;
+    return RecGetReducedSolid (box, in);
+  }
 
-Solid * Solid :: RecGetReducedSolid (const BoxSphere<3> & box, int & in) const
-{
-  Solid * redsol = NULL;
+  Solid * Solid :: RecGetReducedSolid (const BoxSphere<3> & box, INSOLID_TYPE & in) const
+  {
+    Solid * redsol = NULL;
 
-  switch (op)
-    {
-    case TERM:
+    switch (op)
       {
-	/* really old
-	in = surf -> BoxInSolid (box);
-	if (in == 2) // maybe
-	  redsol = new Solid (surf, id);
-	*/
-	in = int(prim -> BoxInSolid (box));
-	if (in == 2) // maybe
-	  redsol = new Solid (prim);
+      case TERM: 
+      case TERM_REF:
+	{
+	  in = prim -> BoxInSolid (box);
+	  if (in == DOES_INTERSECT)
+	    {
+	      redsol = new Solid (prim);
+	      redsol -> op = TERM_REF;
+	    }
+	  break;
+	}
+      case SECTION:
+	{
+	  INSOLID_TYPE in1, in2;
+	  Solid * redsol1, * redsol2;
 
-	//	prim->GetSurface(1).Print (*testout);
+	  redsol1 = s1 -> RecGetReducedSolid (box, in1);
+	  redsol2 = s2 -> RecGetReducedSolid (box, in2);
 
-	break;
-      }
-    case SECTION:
-      {
-	int in1, in2;
-	Solid * redsol1, * redsol2;
+	  if (in1 == IS_OUTSIDE || in2 == IS_OUTSIDE)
+	    {
+	      if (in1 == DOES_INTERSECT) delete redsol1;
+	      if (in2 == DOES_INTERSECT) delete redsol2;
+	      in = IS_OUTSIDE;
+	    }
+	  else
+	    {
+	      if (in1 == DOES_INTERSECT || in2 == DOES_INTERSECT) 
+		in = DOES_INTERSECT;
+	      else 
+		in = IS_INSIDE;
+
+	      if (in1 == DOES_INTERSECT && in2 == DOES_INTERSECT)
+		redsol = new Solid (SECTION, redsol1, redsol2);
+	      else if (in1 == DOES_INTERSECT)
+		redsol = redsol1;
+	      else if (in2 == DOES_INTERSECT)
+		redsol = redsol2;
+	    }
+	  break;
+	}
 
-	redsol1 = s1 -> RecGetReducedSolid (box, in1);
-	redsol2 = s2 -> RecGetReducedSolid (box, in2);
+      case UNION:
+	{
+	  INSOLID_TYPE in1, in2;
+	  Solid * redsol1, * redsol2;
 
-	if (in1 == 0 || in2 == 0)
-	  {
-	    if (in1 == 2) delete redsol1;
-	    if (in2 == 2) delete redsol2;
-	    in = 0;
-	  }
-	else
-	  {
-	    if (in1 == 2 || in2 == 2) in = 2;
-	    else in = 1;
-
-	    if (in1 == 2 && in2 == 2)
-	      redsol = new Solid (SECTION, redsol1, redsol2);
-	    else if (in1 == 2)
-	      redsol = redsol1;
-	    else if (in2 == 2)
-	      redsol = redsol2;
-	  }
-	break;
-      }
+	  redsol1 = s1 -> RecGetReducedSolid (box, in1);
+	  redsol2 = s2 -> RecGetReducedSolid (box, in2);
 
-    case UNION:
-      {
-	int in1, in2;
-	Solid * redsol1, * redsol2;
+	  if (in1 == IS_INSIDE || in2 == IS_INSIDE)
+	    {
+	      if (in1 == DOES_INTERSECT) delete redsol1;
+	      if (in2 == DOES_INTERSECT) delete redsol2;
+	      in = IS_INSIDE;
+	    }
+	  else
+	    {
+	      if (in1 == DOES_INTERSECT || in2 == DOES_INTERSECT) in = DOES_INTERSECT;
+	      else in = IS_OUTSIDE;
+
+	      if (in1 == DOES_INTERSECT && in2 == DOES_INTERSECT)
+		redsol = new Solid (UNION, redsol1, redsol2);
+	      else if (in1 == DOES_INTERSECT)
+		redsol = redsol1;
+	      else if (in2 == DOES_INTERSECT)
+		redsol = redsol2;
+	    }
+	  break;
+	}
 
-	redsol1 = s1 -> RecGetReducedSolid (box, in1);
-	redsol2 = s2 -> RecGetReducedSolid (box, in2);
+      case SUB:
+	{
+	  INSOLID_TYPE in1;
+	  Solid * redsol1 = s1 -> RecGetReducedSolid (box, in1);
 
-	if (in1 == 1 || in2 == 1)
-	  {
-	    if (in1 == 2) delete redsol1;
-	    if (in2 == 2) delete redsol2;
-	    in = 1;
-	  }
-	else
-	  {
-	    if (in1 == 2 || in2 == 2) in = 2;
-	    else in = 0;
-
-	    if (in1 == 2 && in2 == 2)
-	      redsol = new Solid (UNION, redsol1, redsol2);
-	    else if (in1 == 2)
-	      redsol = redsol1;
-	    else if (in2 == 2)
-	      redsol = redsol2;
-	  }
-	break;
-      }
+	  switch (in1)
+	    {
+	    case IS_OUTSIDE: in = IS_INSIDE; break;
+	    case IS_INSIDE:  in = IS_OUTSIDE; break;
+	    case DOES_INTERSECT: in = DOES_INTERSECT; break;
+	    }
 
-    case SUB:
-      {
-	int in1;
-	Solid * redsol1;
-	redsol1 = s1 -> RecGetReducedSolid (box, in1);
-	switch (in1)
-	  {
-	  case 0: in = 1; break;
-	  case 1: in = 0; break;
-	  case 2: in = 2; break;
-	  }
-	if (redsol1)
-	  redsol = new Solid (SUB, redsol1);
-	break;
-      }
+	  if (redsol1)
+	    redsol = new Solid (SUB, redsol1);
+	  break;
+	}
       
-    case ROOT:
-      {
-	int in1;
-	redsol = s1 -> RecGetReducedSolid (box, in1);
-	in = in1;
-	break;
+      case ROOT:
+	{
+	  INSOLID_TYPE in1;
+	  redsol = s1 -> RecGetReducedSolid (box, in1);
+	  in = in1;
+	  break;
+	}
       }
-    }
-  return redsol;
-}
+    return redsol;
+  }
 
 
-int Solid :: NumPrimitives () const
-{
-  switch (op)
-    {
-    case TERM:
-      {
-	return 1;
-      }
-    case UNION:
-    case SECTION:
-      {
-	return s1->NumPrimitives () + s2 -> NumPrimitives();
-      }
-    case SUB:
-    case ROOT:
+  int Solid :: NumPrimitives () const
+  {
+    switch (op)
       {
-	return s1->NumPrimitives ();
+      case TERM: case TERM_REF:
+	{
+	  return 1;
+	}
+      case UNION:
+      case SECTION:
+	{
+	  return s1->NumPrimitives () + s2 -> NumPrimitives();
+	}
+      case SUB:
+      case ROOT:
+	{
+	  return s1->NumPrimitives ();
+	}
       }
-    }
-  return 0;
-}
+    return 0;
+  }
 
-void Solid :: GetSurfaceIndices (ARRAY<int> & surfind) const
-{
-  surfind.SetSize (0);
-  RecGetSurfaceIndices (surfind);
-}
+  void Solid :: GetSurfaceIndices (ARRAY<int> & surfind) const
+  {
+    surfind.SetSize (0);
+    RecGetSurfaceIndices (surfind);
+  }
 
-void Solid :: RecGetSurfaceIndices (ARRAY<int> & surfind) const
-{
-  switch (op)
-    {
-    case TERM:
+  void Solid :: RecGetSurfaceIndices (ARRAY<int> & surfind) const
+  {
+    switch (op)
       {
-	/*
-	int i;
-	for (i = 1; i <= surfind.Size(); i++)
-	  if (surfind.Get(i) == prim->GetSurfaceId())
+      case TERM: case TERM_REF:
+	{
+	  /*
+	    int i;
+	    for (i = 1; i <= surfind.Size(); i++)
+	    if (surfind.Get(i) == prim->GetSurfaceId())
 	    return;
-	surfind.Append (prim->GetSurfaceId());
-	break;
-	*/
-	int i, j;
-	for (j = 0; j < prim->GetNSurfaces(); j++)
-	  if (prim->SurfaceActive (j))
-	    {
-	      bool found = 0;
-	      int siprim = prim->GetSurfaceId(j);
-
-	      for (i = 0; i < surfind.Size(); i++)
-		if (surfind[i] == siprim)
-		  {
-		    found = 1;
-		    break;
-		  }
-	      if (!found) surfind.Append (siprim);
-	    }
-	break;
-      }
-    case UNION:
-    case SECTION:
-      {
-	s1 -> RecGetSurfaceIndices (surfind);
-	s2 -> RecGetSurfaceIndices (surfind);
-	break;
-      }
-    case SUB:
-    case ROOT:
-      {
-	s1 -> RecGetSurfaceIndices (surfind);
-	break;
+	    surfind.Append (prim->GetSurfaceId());
+	    break;
+	  */
+	  for (int j = 0; j < prim->GetNSurfaces(); j++)
+	    if (prim->SurfaceActive (j))
+	      {
+		bool found = 0;
+		int siprim = prim->GetSurfaceId(j);
+
+		for (int i = 0; i < surfind.Size(); i++)
+		  if (surfind[i] == siprim)
+		    {
+		      found = 1;
+		      break;
+		    }
+		if (!found) surfind.Append (siprim);
+	      }
+	  break;
+	}
+      case UNION:
+      case SECTION:
+	{
+	  s1 -> RecGetSurfaceIndices (surfind);
+	  s2 -> RecGetSurfaceIndices (surfind);
+	  break;
+	}
+      case SUB:
+      case ROOT:
+	{
+	  s1 -> RecGetSurfaceIndices (surfind);
+	  break;
+	}
       }
-    }
-}
+  }
 
 
 
-void Solid :: GetSurfaceIndices (IndexSet & iset) const
-{
-  iset.Clear();
-  RecGetSurfaceIndices (iset);
-}
+  void Solid :: GetSurfaceIndices (IndexSet & iset) const
+  {
+    iset.Clear();
+    RecGetSurfaceIndices (iset);
+  }
 
-void Solid :: RecGetSurfaceIndices (IndexSet & iset) const
-{
-  switch (op)
-    {
-    case TERM:
+  void Solid :: RecGetSurfaceIndices (IndexSet & iset) const
+  {
+    switch (op)
       {
-	/*
-	int i;
-	for (i = 1; i <= surfind.Size(); i++)
-	  if (surfind.Get(i) == prim->GetSurfaceId())
+      case TERM: case TERM_REF:
+	{
+	  /*
+	    int i;
+	    for (i = 1; i <= surfind.Size(); i++)
+	    if (surfind.Get(i) == prim->GetSurfaceId())
 	    return;
-	surfind.Append (prim->GetSurfaceId());
-	break;
-	*/
-	int i, j;
-	for (j = 0; j < prim->GetNSurfaces(); j++)
-	  if (prim->SurfaceActive (j))
-	    {
-	      int siprim = prim->GetSurfaceId(j);
-	      iset.Add (siprim);
-	    }
-	break;
-      }
-    case UNION:
-    case SECTION:
-      {
-	s1 -> RecGetSurfaceIndices (iset);
-	s2 -> RecGetSurfaceIndices (iset);
-	break;
-      }
-    case SUB:
-    case ROOT:
-      {
-	s1 -> RecGetSurfaceIndices (iset);
-	break;
+	    surfind.Append (prim->GetSurfaceId());
+	    break;
+	  */
+	  for (int j = 0; j < prim->GetNSurfaces(); j++)
+	    if (prim->SurfaceActive (j))
+	      {
+		int siprim = prim->GetSurfaceId(j);
+		iset.Add (siprim);
+	      }
+	  break;
+	}
+      case UNION:
+      case SECTION:
+	{
+	  s1 -> RecGetSurfaceIndices (iset);
+	  s2 -> RecGetSurfaceIndices (iset);
+	  break;
+	}
+      case SUB:
+      case ROOT:
+	{
+	  s1 -> RecGetSurfaceIndices (iset);
+	  break;
+	}
       }
-    }
-}
-
-
-
-
-
-
-BlockAllocator Solid :: ball(sizeof (Solid));
-
-
-void * Solid :: operator new(size_t s)
-{
-  return ball.Alloc();
-}
-
-void Solid :: operator delete (void * p)
-{
-  ball.Free (p);
-}
-
+  }
 
 
 
 
 
-ReducePrimitiveIterator :: 
-ReducePrimitiveIterator (const BoxSphere<3> & abox)
-  : SolidIterator(), box(abox)
-{
-  ;
-}
-
-ReducePrimitiveIterator :: 
-~ReducePrimitiveIterator ()
-{
-  ;
-}
-
-void 
-ReducePrimitiveIterator :: Do (Solid * sol)
-{
-  if (sol -> GetPrimitive())
-    sol -> GetPrimitive() -> Reduce (box);
-}
-
-
-
-
-UnReducePrimitiveIterator :: 
-UnReducePrimitiveIterator ()
-  : SolidIterator()
-{
-  ;
-}
-
-UnReducePrimitiveIterator :: 
-~UnReducePrimitiveIterator ()
-{
-  ;
-}
-
-void 
-UnReducePrimitiveIterator :: Do (Solid * sol)
-{
-  if (sol -> GetPrimitive())
-    sol -> GetPrimitive() -> UnReduce ();
-}
+  BlockAllocator Solid :: ball(sizeof (Solid));
 }
diff --git a/Netgen/libsrc/csg/solid.hpp b/Netgen/libsrc/csg/solid.hpp
index b09d8fd682..796729125a 100644
--- a/Netgen/libsrc/csg/solid.hpp
+++ b/Netgen/libsrc/csg/solid.hpp
@@ -21,8 +21,8 @@ class Solid;
 class SolidIterator
 {
 public:
-  SolidIterator ();
-  virtual ~SolidIterator ();
+  SolidIterator () { ; }
+  virtual ~SolidIterator () { ; }
   virtual void Do (Solid * sol) = 0;
 };
 
@@ -32,7 +32,7 @@ class Solid
 {
 public:
   
-  typedef enum optyp1 { TERM, SECTION, UNION, SUB, ROOT, DUMMY } optyp;
+  typedef enum optyp1 { TERM, TERM_REF, SECTION, UNION, SUB, ROOT, DUMMY } optyp;
   
 private:
   char * name;
@@ -40,15 +40,15 @@ private:
   Solid * s1, * s2;
   
   optyp op;
-  int visited;
+  bool visited;
   double maxh;
 
-  static int cntnames;
+  // static int cntnames;
 
 public:
   Solid (Primitive * aprim);
   Solid (optyp aop, Solid * as1, Solid * as2 = NULL);
-  virtual ~Solid ();
+  ~Solid ();
 
   const char * Name () const { return name; }
   void SetName (const char * aname);
@@ -57,7 +57,7 @@ public:
   void Transform (Transformation<3> & trans);
 
   
-  void IterateSolid (SolidIterator & it, int only_once = 0);
+  void IterateSolid (SolidIterator & it, bool only_once = 0);
 
   
   void Boundaries (const Point<3> & p, ARRAY<int> & bounds) const;
@@ -66,21 +66,24 @@ public:
   void GetSurfaceIndices (IndexSet & iset) const;
 
   Primitive * GetPrimitive ()
-    { return (op == TERM) ? prim : NULL; }
+    { return (op == TERM || op == TERM_REF) ? prim : NULL; }
   const Primitive * GetPrimitive () const
-    { return (op == TERM) ? prim : NULL; }
+    { return (op == TERM || op == TERM_REF) ? prim : NULL; }
+
+  Solid * S1() { return s1; }
+  Solid * S2() { return s2; }
 
   // geometric tests
 
-  int IsIn (const Point<3> & p, double eps = 1e-6) const;
-  int IsStrictIn (const Point<3> & p, double eps = 1e-6) const;
-  int VectorIn (const Point<3> & p, const Vec<3> & v, double eps = 1e-6) const;
-  int VectorStrictIn (const Point<3> & p, const Vec<3> & v, double eps = 1e-6) const;
+  bool IsIn (const Point<3> & p, double eps = 1e-6) const;
+  bool IsStrictIn (const Point<3> & p, double eps = 1e-6) const;
+  bool VectorIn (const Point<3> & p, const Vec<3> & v, double eps = 1e-6) const;
+  bool VectorStrictIn (const Point<3> & p, const Vec<3> & v, double eps = 1e-6) const;
   
-  int VectorIn2 (const Point<3> & p, const Vec<3> & v1, const Vec<3> & v2,
-		 double eps = 1e-6) const;
-  int VectorIn2Rec (const Point<3> & p, const Vec<3> & v1, const Vec<3> & v2,
-		    double eps = 1e-6) const;
+  bool VectorIn2 (const Point<3> & p, const Vec<3> & v1, const Vec<3> & v2,
+		  double eps = 1e-6) const;
+  bool VectorIn2Rec (const Point<3> & p, const Vec<3> & v1, const Vec<3> & v2,
+		     double eps = 1e-6) const;
 
 
   ///
@@ -93,7 +96,7 @@ public:
   ///
   int OnFace (const Point<3> & p, const Vec<3> & v) const;
   ///
-  virtual void Print (ostream & str) const;
+  void Print (ostream & str) const;
   ///
   void CalcSurfaceInverse ();
   ///
@@ -110,8 +113,16 @@ public:
 
 
   static BlockAllocator ball;
-  void * operator new(size_t);
-  void operator delete (void *);
+  void * operator new(size_t /* s */) 
+  {
+    return ball.Alloc();
+  }
+
+  void operator delete (void * p)
+  {
+    ball.Free (p);
+  }
+
 
 protected:
   ///
@@ -130,7 +141,7 @@ protected:
   ///
   void CalcSurfaceInverseRec (int inv);
   ///
-  Solid * RecGetReducedSolid (const BoxSphere<3> & box, int & in) const;
+  Solid * RecGetReducedSolid (const BoxSphere<3> & box, INSOLID_TYPE & in) const;
   ///
   void RecGetSurfaceIndices (ARRAY<int> & surfind) const;
   void RecGetSurfaceIndices (IndexSet & iset) const;
@@ -142,22 +153,43 @@ protected:
 };
 
 
+inline ostream & operator<< (ostream & ost, const Solid & sol)
+{
+  sol.Print (ost);
+  return ost;
+}
+
+
+
+
+
+
 class ReducePrimitiveIterator : public SolidIterator
 {
   const BoxSphere<3> & box;
 public:
-  ReducePrimitiveIterator (const BoxSphere<3> & abox);
-  virtual ~ReducePrimitiveIterator ();
-  virtual void Do (Solid * sol);  
+  ReducePrimitiveIterator (const BoxSphere<3> & abox)
+    : SolidIterator(), box(abox) { ; }
+  virtual ~ReducePrimitiveIterator () { ; }
+  virtual void Do (Solid * sol)
+  {
+    if (sol -> GetPrimitive())
+      sol -> GetPrimitive() -> Reduce (box);
+  }
 };
 
 
 class UnReducePrimitiveIterator : public SolidIterator
 {
 public:
-  UnReducePrimitiveIterator ();
-  virtual ~UnReducePrimitiveIterator ();
-  virtual void Do (Solid * sol);  
+  UnReducePrimitiveIterator () { ; }
+  virtual ~UnReducePrimitiveIterator () { ; }
+  virtual void Do (Solid * sol)
+  {
+    if (sol -> GetPrimitive())
+      sol -> GetPrimitive() -> UnReduce ();
+  }
 };
 
+
 #endif
diff --git a/Netgen/libsrc/csg/specpoin.cpp b/Netgen/libsrc/csg/specpoin.cpp
index 12fe79966a..c9b2477bb0 100644
--- a/Netgen/libsrc/csg/specpoin.cpp
+++ b/Netgen/libsrc/csg/specpoin.cpp
@@ -4,1366 +4,1228 @@
 
 
 /*
-
-   Special Point calculation uses the global Flags:
-
-
-   size .. 500       cube = [-size, size]^3
-   relydegtest       when to rely on degeneration ?
-   calccp            calculate points of intersection ?
-   cpeps1            eps for degenerated poi
-   calcep            calculate points of extreme coordinates ?
-   epeps1            eps for degenerated edge
-   epeps2            eps for axis parallel pec
-   epspointdist      eps for distance of special points 
+  Special Point calculation uses the global Flags:
+
+  relydegtest       when to rely on degeneration ?
+  calccp            calculate points of intersection ?
+  cpeps1            eps for degenerated poi
+  calcep            calculate points of extreme coordinates ?
+  epeps1            eps for degenerated edge
+  epeps2            eps for axis parallel pec
+  epspointdist      eps for distance of special points 
 */
 
 
 namespace netgen
 {
-void ProjectToEdge (const Surface * f1, const Surface * f2, Point<3> & hp);
+  void ProjectToEdge (const Surface * f1, const Surface * f2, Point<3> & hp);
 
 
 
-  /*
-SpecialPoint :: SpecialPoint ()
-  : p(), v(), layer(0)
-{
-  ;
-}
-  */
+  SpecialPoint :: SpecialPoint (const SpecialPoint & sp)
+  {
+    p = sp.p;
+    v = sp.v;
+    s1 = sp.s1;
+    s2 = sp.s2;
+    layer = sp.layer;
+    unconditional = sp.unconditional;
+  }
+  
+  SpecialPoint & SpecialPoint :: operator= (const SpecialPoint & sp)
+  {
+    p = sp.p;
+    v = sp.v;
+    s1 = sp.s1;
+    s2 = sp.s2;
+    layer = sp.layer;
+    unconditional = sp.unconditional;
+    return *this;
+  }
 
 
-SpecialPoint :: SpecialPoint (const SpecialPoint & sp)
-{
-  p = sp.p;
-  v = sp.v;
-  s1 = sp.s1;
-  s2 = sp.s2;
-  layer = sp.layer;
-  unconditional = sp.unconditional;
-}
-  
-SpecialPoint & SpecialPoint :: operator= (const SpecialPoint & sp)
-{
-  p = sp.p;
-  v = sp.v;
-  s1 = sp.s1;
-  s2 = sp.s2;
-  layer = sp.layer;
-  unconditional = sp.unconditional;
-  return *this;
-}
+  void SpecialPoint :: Print (ostream & str)
+  {
+    str << "p = " << p << "   v = " << v 
+	<< " s1/s2 = " << s1 << "/" << s2
+	<< " layer = " << layer
+	<< endl;
+  }
 
-bool SpecialPoint :: HasSurfaces (int as1, int as2) const
-{
-  return (s1 == as1 && s2 == as2 || s1 == as2 && s2 == as1);
-}
 
-void SpecialPoint :: Print (ostream & str)
-{
-  str << "p = " << p << "   v = " << v 
-      << " s1/s2 = " << s1 << "/" << s2
-      << " layer = " << layer
-      << endl;
-}
+  static ARRAY<int> numprim_hist;
 
+  SpecialPointCalculation :: SpecialPointCalculation ()
+  {
+    ;  
+  }
 
+  void SpecialPointCalculation :: 
+  CalcSpecialPoints (const CSGeometry & ageometry, 
+		     ARRAY<MeshPoint> & apoints)
+  {
+    geometry = &ageometry;
+    points = &apoints;
 
+    size = geometry->MaxSize(); 
+    (*testout) << "Find Special Points" << endl;
+    (*testout) << "maxsize = " << size << endl;
 
-SpecialPointCalculation :: SpecialPointCalculation ()
-{
-  ;  
-}
+    cpeps1 = 1e-6; 
+    epeps1 = 1e-3; 
+    epeps2 = 1e-6; 
 
-void SpecialPointCalculation :: 
-CalcSpecialPoints (const CSGeometry & ageometry, 
-		   ARRAY<MeshPoint> & apoints)
-{
-  int i;
+    epspointdist2 = sqr (size * 1e-8); 
+    relydegtest = size * 1e-4; 
 
-  geometry = &ageometry;
-  points = &apoints;
 
-  size = geometry->MaxSize();  // globflags.GetNumFlag ("maxsize", 500);
-  (*testout) << "Find Special Points" << endl;
-  (*testout) << "maxsize = " << size << endl;
+    BoxSphere<3> box (Point<3> (-size, -size, -size),
+		      Point<3> ( size,  size,  size));
 
-  cpeps1 = 1e-6; 
-  epeps1 = 1e-3; 
-  epeps2 = 1e-6; 
+    box.CalcDiamCenter();
+    PrintMessage (3, "main-solids: ", geometry->GetNTopLevelObjects());
 
-  epspointdist2 = sqr (size * 1e-8); 
-  relydegtest = size * 1e-4; 
+    numprim_hist.SetSize (geometry->GetNSurf()+1);
+    numprim_hist = 0;
 
+    for (int i = 0; i < geometry->GetNTopLevelObjects(); i++)
+      {
+	const TopLevelObject * tlo = geometry->GetTopLevelObject(i);
 
-  BoxSphere<3> box (Point<3> (-size, -size, -size),
-		    Point<3> ( size,  size,  size));
-  box.CalcDiamCenter();
-  PrintMessage (3, "main-solids: ", geometry->GetNTopLevelObjects());
+	(*testout) << "tlo " << i << ":" << endl
+		   << *tlo->GetSolid() << endl;
 
-  for (i = 0; i < geometry->GetNTopLevelObjects(); i++)
-    {
-      (*testout) << "tlo " << i << ":" << endl;
-      const TopLevelObject * tlo = geometry->GetTopLevelObject(i);
-      tlo->GetSolid()->Print (*testout); 
-      (*testout) << endl;
-      CalcSpecialPointsRec (tlo->GetSolid(), tlo->GetLayer(),
-			    box, 1, 1, 1);
-    }
+	CalcSpecialPointsRec (tlo->GetSolid(), tlo->GetLayer(),
+			      box, 1, 1, 1);
+      }
   
-  PrintDot ('\n');
+    // add user point:
+    for (int i = 0; i < geometry->GetNUserPoints(); i++)
+      AddPoint (geometry->GetUserPoint(i), 1);
 
+    PrintMessage (3, "Found points ", apoints.Size());
 
-  // add user point:
-  int found = 0;
-  for (i = 0; i < geometry->GetNUserPoints(); i++)
-    AddPoint (geometry->GetUserPoint(i), 1);
+    for (int i = 0; i < boxesinlevel.Size(); i++)
+      (*testout) << "level " << i << " has " 
+		 << boxesinlevel[i] << " boxes" << endl;
+    (*testout) << "numprim_histogramm = " << endl << numprim_hist << endl;
+  }
   
-  PrintMessage (3, apoints.Size(), " special points");
 
-  for (i = 0; i < boxesinlevel.Size(); i++)
-    (*testout) << "level " << i << " has " 
-	       << boxesinlevel[i] << " boxes" << endl;
-}
 
+  void SpecialPointCalculation :: 
+  CalcSpecialPointsRec (const Solid * sol, int layer,
+			const BoxSphere<3> & box, 
+			int level, bool calccp, bool calcep)
+  {
+    if (multithread.terminate)
+      throw NgException ("Meshing stopped");
 
 
-int debug;
-void SpecialPointCalculation :: 
-CalcSpecialPointsRec (const Solid * sol, int layer,
-		      const BoxSphere<3> & box, 
-		      int level, bool calccp, bool calcep)
-{
-  if (multithread.terminate)
-    return;
+    if (!sol) return;
 
-  int i;
-  BoxSphere<3> sbox;
-  Solid * redsol;
+    if (level >= 100)
+      {
+	MyStr err =
+	  MyStr("Problems in CalcSpecialPoints\nPoint: ") + MyStr (box.Center());
+	throw NgException (err.c_str());
+      }
 
-  int numprim;
 
-  bool decision;
-  bool possiblecrossp, possibleexp;  // possible cross or extremalpoint
-  bool surecrossp, sureexp;          // sure ...
+    bool decision;
+    bool possiblecrossp, possibleexp;  // possible cross or extremalpoint
+    bool surecrossp, sureexp;          // sure ...
   
-  static ARRAY<int> locsurf;  // attention: array is static
-
-
-  Point<3> p;
-  int k1, k2, k3;
-  int extremdir;
-  double hd;
-
-  if (!sol) return;
-
-
-  if (level >= 100)
-    {
-      cerr << "Problems in CalcSpecialPoints" << endl;
-      cerr << "Point: " << box.Center() << endl;
-      exit (1);
-    }
+    static ARRAY<int> locsurf;  // attention: array is static
 
-  static int cntbox = 0;
-  cntbox++;
-  if (cntbox % 10000 == 0) 
-    PrintDot ();
+    static int cntbox = 0;
+    cntbox++;
 
-  if (level <= boxesinlevel.Size())
-    boxesinlevel.Elem(level)++;
-  else
-    boxesinlevel.Append (1);
+    if (level <= boxesinlevel.Size())
+      boxesinlevel.Elem(level)++;
+    else
+      boxesinlevel.Append (1);
 
-  /*
-  numprim = sol -> NumPrimitives();
-  sol -> GetSurfaceIndices (locsurf);
-  */
+    /*
+      numprim = sol -> NumPrimitives();
+      sol -> GetSurfaceIndices (locsurf);
+    */
 
-  debug = 0;
-  // box.IsIn (Point<3> (4.9, 1.279, 2.8));
+    geometry -> GetIndependentSurfaceIndices (sol, box, locsurf);
+    int numprim = locsurf.Size();
 
+    numprim_hist[numprim]++;
 
-  geometry -> GetIndependentSurfaceIndices (sol, box, locsurf);
-  numprim = locsurf.Size();
-  
-  if (debug)
-    {
-      (*testout) << "box = " << box.PMin() << "-" << box.PMax()
-		 << " np = " << numprim << " : ";
-      for (i = 1; i <= locsurf.Size(); i++)
-	(*testout) << " " << locsurf.Get(i);
-      (*testout) << " diam = " << box.Diam();
-      (*testout) << " numprim = " << numprim;
-      (*testout) << endl;
-    }
-
-  p = box.Center();
-  
-  /*
-  (*testout) << "box = " << box.PMin() << " - " << box.PMax()
-	     << ", lev = " << level 
-	     << ", nprim = " << sol->NumPrimitives() 
-	     << ", lsurf = " << locsurf.Size() << endl;
-
-  for (i = 1; i <= locsurf.Size(); i++)
-    geometry->GetSurface (locsurf.Get(i)) -> Print (*testout);
-  sol -> Print (*testout);
-  */
+    Point<3> p = box.Center();
 
-  /*
-    for (i = 1; i <= locsurf.Size(); i++)
-    (*testout) << locsurf.Get(i) << " ";
-    (*testout) << "C = " << box.Center() << " diam = " << box.Diam() << endl;
-    */
 
-  possiblecrossp = (numprim >= 3) && calccp;
-  surecrossp = 0;
+    possiblecrossp = (numprim >= 3) && calccp;
+    surecrossp = 0;
 
-  if (possiblecrossp && (locsurf.Size() <= 10))
-    {
-      decision = 1;
-      surecrossp = 0;
+    if (possiblecrossp && (locsurf.Size() <= 5 || level > 50))
+      {
+	decision = 1;
+	surecrossp = 0;
 
-      for (k1 = 1; k1 <= locsurf.Size() - 2; k1++)
-	for (k2 = k1 + 1; k2 <= locsurf.Size() - 1; k2++)
-	  for (k3 = k2 + 1; k3 <= locsurf.Size(); k3++)
-	    {
-	      int nc, deg;
-	      nc = CrossPointNewtonConvergence 
-		(geometry->GetSurface(locsurf.Get(k1)), 
-		 geometry->GetSurface(locsurf.Get(k2)), 
-		 geometry->GetSurface(locsurf.Get(k3)), p );
+	for (int k1 = 1; k1 <= locsurf.Size() - 2; k1++)
+	  for (int k2 = k1 + 1; k2 <= locsurf.Size() - 1; k2++)
+	    for (int k3 = k2 + 1; k3 <= locsurf.Size(); k3++)
+	      {
+		int nc, deg;
+		nc = CrossPointNewtonConvergence 
+		  (geometry->GetSurface(locsurf.Get(k1)), 
+		   geometry->GetSurface(locsurf.Get(k2)), 
+		   geometry->GetSurface(locsurf.Get(k3)), p );
 	      
-	      deg = CrossPointDegenerated 
-		(geometry->GetSurface(locsurf.Get(k1)), 
-		 geometry->GetSurface(locsurf.Get(k2)), 
-		 geometry->GetSurface(locsurf.Get(k3)), box );
+		deg = CrossPointDegenerated 
+		  (geometry->GetSurface(locsurf.Get(k1)), 
+		   geometry->GetSurface(locsurf.Get(k2)), 
+		   geometry->GetSurface(locsurf.Get(k3)), box );
 	      
-	      if (!nc && !deg) decision = 0;
-	      if (nc) surecrossp = 1;
-	    }
+		if (!nc && !deg) decision = 0;
+		if (nc) surecrossp = 1;
+	      }
 
-      if (decision && surecrossp)
-	{
-	  for (k1 = 1; k1 <= locsurf.Size() - 2; k1++)
-	    for (k2 = k1 + 1; k2 <= locsurf.Size() - 1; k2++)
-	      for (k3 = k2 + 1; k3 <= locsurf.Size(); k3++)
-		{
-		  if (CrossPointNewtonConvergence 
-		      (geometry->GetSurface(locsurf.Get(k1)), 
-		       geometry->GetSurface(locsurf.Get(k2)), 
-		       geometry->GetSurface(locsurf.Get(k3)), p ) )
-		    {
-		      Point<3> pp = p;
-		      CrossPointNewton 
+	if (decision && surecrossp)
+	  {
+	    for (int k1 = 1; k1 <= locsurf.Size() - 2; k1++)
+	      for (int k2 = k1 + 1; k2 <= locsurf.Size() - 1; k2++)
+		for (int k3 = k2 + 1; k3 <= locsurf.Size(); k3++)
+		  {
+		    if (CrossPointNewtonConvergence 
 			(geometry->GetSurface(locsurf.Get(k1)), 
 			 geometry->GetSurface(locsurf.Get(k2)), 
-			 geometry->GetSurface(locsurf.Get(k3)), pp);
-              
-		      BoxSphere<3> hbox (pp, pp);
-		      hbox.Increase (1e-8);
-
-		      if (pp(0) > box.PMin()(0) - 1e-5 && 
-			  pp(0) < box.PMax()(0) + 1e-5 &&
-			  pp(1) > box.PMin()(1) - 1e-5 && 
-			  pp(1) < box.PMax()(1) + 1e-5 &&
-			  pp(2) > box.PMin()(2) - 1e-5 && 
-			  pp(2) < box.PMax()(2) + 1e-5 &&
-			  sol -> IsIn (pp) && !sol->IsStrictIn (pp) &&
-			  !CrossPointDegenerated
+			 geometry->GetSurface(locsurf.Get(k3)), p ) )
+		      {
+			Point<3> pp = p;
+			CrossPointNewton 
 			  (geometry->GetSurface(locsurf.Get(k1)), 
 			   geometry->GetSurface(locsurf.Get(k2)), 
-			   geometry->GetSurface(locsurf.Get(k3)), hbox ))
-
-			{ 
-			  //                AddCrossPoint (locsurf, sol, p);
-			  BoxSphere<3> boxp (pp, pp);
-			  boxp.Increase (1e-3);
-			  boxp.CalcDiamCenter();
-			  ARRAY<int> locsurf2;
-
-			  geometry -> GetIndependentSurfaceIndices (sol, boxp, locsurf2);
-			  
-			  /*
-			  ReducePrimitiveIterator rpi(boxp);
-			  UnReducePrimitiveIterator urpi;
-			  
-			  ((Solid*)sol) -> IterateSolid (rpi);
-			  sol -> GetIndependentSurfaceIndices (locsurf2);
-			  ((Solid*)sol) -> IterateSolid (urpi);
-			  */
-			  bool found1 = 0, found2 = 0, found3 = 0;
-			  for (i = 1; i <= locsurf2.Size(); i++)
-			    {
-			      if (locsurf2.Get(i) == locsurf.Get(k1))
-				found1 = 1;
-			      if (locsurf2.Get(i) == locsurf.Get(k2))
-				found2 = 1;
-			      if (locsurf2.Get(i) == locsurf.Get(k3))
-				found3 = 1;
-			    }
+			   geometry->GetSurface(locsurf.Get(k3)), pp);
+              
+			BoxSphere<3> hbox (pp, pp);
+			hbox.Increase (1e-8);
+
+			if (pp(0) > box.PMin()(0) - 1e-5 && 
+			    pp(0) < box.PMax()(0) + 1e-5 &&
+			    pp(1) > box.PMin()(1) - 1e-5 && 
+			    pp(1) < box.PMax()(1) + 1e-5 &&
+			    pp(2) > box.PMin()(2) - 1e-5 && 
+			    pp(2) < box.PMax()(2) + 1e-5 &&
+			    sol -> IsIn (pp) && !sol->IsStrictIn (pp) &&
+			    !CrossPointDegenerated
+			    (geometry->GetSurface(locsurf.Get(k1)), 
+			     geometry->GetSurface(locsurf.Get(k2)), 
+			     geometry->GetSurface(locsurf.Get(k3)), hbox ))
+
+			  { 
+			    //                AddCrossPoint (locsurf, sol, p);
+			    BoxSphere<3> boxp (pp, pp);
+			    boxp.Increase (1e-3);
+			    boxp.CalcDiamCenter();
+			    ARRAY<int> locsurf2;
+
+			    geometry -> GetIndependentSurfaceIndices (sol, boxp, locsurf2);
 			  
-			  if (found1 && found2 && found3)
-			    if (AddPoint (pp, layer))
+			    bool found1 = 0, found2 = 0, found3 = 0;
+			    for (int i = 0; i < locsurf2.Size(); i++)
 			      {
-				(*testout) << "Crosspoint found: " << pp 
-					   << " diam = " << box.Diam() << endl;
-				(*testout) << "surfs: " 
-					   << locsurf.Get(k1) << "," 
-					   << locsurf.Get(k2) << "," 
-					   << locsurf.Get(k3) << endl;
+				if (locsurf2[i] == locsurf.Get(k1)) found1 = 1;
+				if (locsurf2[i] == locsurf.Get(k2)) found2 = 1;
+				if (locsurf2[i] == locsurf.Get(k3)) found3 = 1;
 			      }
-			}
-		    }
-		}
-	}
+
+			    if (found1 && found2 && found3)
+			      if (AddPoint (pp, layer))
+				{
+				  (*testout) << "Crosspoint found: " << pp 
+					     << " diam = " << box.Diam()
+					     << ",  surfs: " 
+					     << locsurf.Get(k1) << "," 
+					     << locsurf.Get(k2) << "," 
+					     << locsurf.Get(k3) << endl;
+				}
+			  }
+		      }
+		  }
+	  }
       
-      if (decision)
-	possiblecrossp = 0;
-    }
+	if (decision)
+	  possiblecrossp = 0;
+      }
 
 
 
 
-  possibleexp = (numprim >= 2) && calcep;
-    
-  if (possibleexp && (locsurf.Size() <= 10))
-    {
-      decision = 1;
-      sureexp = 0;
+    possibleexp = (numprim >= 2) && calcep;
 
-      for (k1 = 1; k1 <= locsurf.Size() - 1; k1++)
-	for (k2 = k1+1; k2 <= locsurf.Size(); k2++)
-	  {
-	    bool nc, deg;
-	    
-	    nc = EdgeNewtonConvergence 
-	      (geometry->GetSurface(locsurf.Get(k1)),
-	       geometry->GetSurface(locsurf.Get(k2)),
-	       p);
-	    
-	    deg = EdgeDegenerated 
-	      (geometry->GetSurface(locsurf.Get(k1)),
-	       geometry->GetSurface(locsurf.Get(k2)),
-	       box);
-	    
-	    if (!nc && !deg) decision = 0;
-	    if (nc) sureexp = 1;
 
-	    if (debug)
+    if (numprim == 2)
+      {
+	const Surface * surf1 = geometry->GetSurface(locsurf[0]);
+	const Surface * surf2 = geometry->GetSurface(locsurf[1]);
+	
+	const Plane * plane1 = dynamic_cast<const Plane*> (surf1);
+	const Plane * plane2 = dynamic_cast<const Plane*> (surf2);
+	const QuadraticSurface * quadric1 = dynamic_cast<const QuadraticSurface*> (surf1);
+	const QuadraticSurface * quadric2 = dynamic_cast<const QuadraticSurface*> (surf2);
+
+	if (plane1 && plane2)
+	  possibleexp = 0;
+	else
+	  {
+	    ARRAY<Point<3> > pts;
+	    if (plane1 && quadric2)
 	      {
-		(*testout) << "p = " << p << " s1,2 = " << locsurf.Get(k1) << ", " << locsurf.Get(k2) 
-			   << " nc = " << nc << " deg = " << deg << endl;
-	      }
+		ComputeExtremalPoints (plane1, quadric2, pts);
+		possibleexp = 0;
+	      }		
+	    else if (plane2 && quadric1)
+	      {
+		ComputeExtremalPoints (plane2, quadric1, pts);
+		possibleexp = 0;
+	      }		
+
+	    for (int j = 0; j < pts.Size(); j++)
+	      if (Dist (pts[j], box.Center()) < box.Diam()/2 &&
+		  sol -> IsIn (pts[j]) && !sol->IsStrictIn (pts[j]) )
+		{
+		  if (AddPoint (pts[j], layer))
+		    (*testout) << "Extremal point found: " << pts[j] << endl;
+		}  
 	  }
+      }
 
-      if (decision && sureexp)
-	{
-	  for (k1 = 1; k1 <= locsurf.Size() - 1; k1++)
-	    for (k2 = k1+1; k2 <= locsurf.Size(); k2++)
-	      {
-		if (
-		    EdgeNewtonConvergence 
-		    (geometry->GetSurface(locsurf.Get(k1)),
-		     geometry->GetSurface(locsurf.Get(k2)),
-		     p) )
-		  {
-		    EdgeNewton 
-		      (geometry->GetSurface(locsurf.Get(k1)),
-		       geometry->GetSurface(locsurf.Get(k2)),
-		       p);
+    
+    if (possibleexp && (locsurf.Size() <= 5 || level >= 50))
+      {
+	decision = 1;
+	sureexp = 0;
+
+	/*
+	(*testout) << "extremal surfs = ";
+	for (int k5 = 0; k5 < locsurf.Size(); k5++)
+	  (*testout) << typeid(*geometry->GetSurface(locsurf[k5])).name() << " ";
+	(*testout) << "\n";
+	*/
+
+	for (int k1 = 0; k1 < locsurf.Size() - 1; k1++)
+	  for (int k2 = k1+1; k2 < locsurf.Size(); k2++)
+	    {
+	      const Surface * surf1 = geometry->GetSurface(locsurf[k1]);
+	      const Surface * surf2 = geometry->GetSurface(locsurf[k2]);
+	      /*
+	      (*testout) << "edgecheck, types = " << typeid(*surf1).name() << ", " << typeid(*surf2).name()
+			 << "edge-newton-conv = " << EdgeNewtonConvergence (surf1, surf2, p)
+			 << "edge-deg = " << EdgeDegenerated (surf1, surf2, box)
+			 << "\n";
+	      */
+	      if (EdgeNewtonConvergence (surf1, surf2, p) ) 
+		sureexp = 1;
+	      else
+		{
+		  if (!EdgeDegenerated (surf1, surf2, box)) 
+		    decision = 0;
+		}
+	    }
+
+	if (decision && sureexp)
+	  {
+	    for (int k1 = 0; k1 < locsurf.Size() - 1; k1++)
+	      for (int k2 = k1+1; k2 < locsurf.Size(); k2++)
+		{
+		  const Surface * surf1 = geometry->GetSurface(locsurf[k1]);
+		  const Surface * surf2 = geometry->GetSurface(locsurf[k2]);
+
+		  if (EdgeNewtonConvergence (surf1, surf2, p))
+		    {
+		      EdgeNewton (surf1, surf2, p);
 		    
-		    Point<3> pp;
-		    if (IsEdgeExtremalPoint 
-		      (geometry->GetSurface(locsurf.Get(k1)),
-		       geometry->GetSurface(locsurf.Get(k2)),
-		       p, pp, box.Diam()/2))
+		      Point<3> pp;
+		      if (IsEdgeExtremalPoint (surf1, surf2, p, pp, box.Diam()/2))
 			{
-			  (*testout) << "extremalpoint (nearly) found:" 
-				     << pp
-				     << endl;
+			  (*testout) << "extremalpoint (nearly) found:" << pp << endl;
+
 			  if (Dist (pp, box.Center()) < box.Diam()/2 &&
 			      sol -> IsIn (pp) && !sol->IsStrictIn (pp) )
 			    {
-			      //                AddExtremalPoint (locsurf.Get(k1), locsurf.Get(k2), p);
 			      if (AddPoint (pp, layer))
 				(*testout) << "Extremal point found: " << pp << endl;
 			    }  
 			}            
-		  }
-	      }
-	}
-      if (decision)
-	possibleexp = 0;
-    }
+		    }
+		}
+	  }
+	if (decision)
+	  possibleexp = 0;
+      }
  
 
 
-  if (possiblecrossp || possibleexp)
-    {
-      for (i = 0; i < 8; i++)
-	{
-	  box.GetSubBox (i, sbox);
-	  sbox.Increase (1e-4 * sbox.Diam());
+    if (possiblecrossp || possibleexp)
+      {
+	BoxSphere<3> sbox;
+	for (int i = 0; i < 8; i++)
+	  {
+	    box.GetSubBox (i, sbox);
+	    sbox.Increase (1e-4 * sbox.Diam());
 
-	  redsol = sol -> GetReducedSolid (sbox);
-
-	  if (redsol)
-	    {
-	      CalcSpecialPointsRec (redsol, layer, sbox, level+1, calccp, calcep);
-	      delete redsol;
-	    }
-	}
-    }
-}
+	    Solid * redsol = sol -> GetReducedSolid (sbox);
 
+	    if (redsol)
+	      {
+		CalcSpecialPointsRec (redsol, layer, sbox, level+1, calccp, calcep);
+		delete redsol;
+	      }
+	  }
+      }
+  }
 
 
 
 
-/******* Tests for Point of intersection **********************/
 
+  /******* Tests for Point of intersection **********************/
 
 
-bool SpecialPointCalculation :: 
-CrossPointNewtonConvergence (const Surface * f1, 
-			     const Surface * f2, 
-			     const Surface * f3,
-			     const Point<3> & p)
-{
-  int i;
-  Vec<3> grad;
-  Vec<3> rs, x;
-  Mat<3> jacobi, inv;
-  double alpha, beta, gamma, eta;
-  double sum;
-  int j;
-
-
-  rs(0) = f1->CalcFunctionValue (p);
-  rs(1) = f2->CalcFunctionValue (p);
-  rs(2) = f3->CalcFunctionValue (p);
-
-  f1->CalcGradient (p, grad);
-  jacobi(0,0) = grad(0);
-  jacobi(0,1) = grad(1);
-  jacobi(0,2) = grad(2);
-
-  f2->CalcGradient (p, grad);
-  jacobi(1,0) = grad(0);
-  jacobi(1,1) = grad(1);
-  jacobi(1,2) = grad(2);
-
-  f3->CalcGradient (p, grad);
-  jacobi(2,0) = grad(0);
-  jacobi(2,1) = grad(1);
-  jacobi(2,2) = grad(2);
-
-  alpha = 1;
-  if (fabs (Det (jacobi)) > 1e-8)
-    {
-      CalcInverse (jacobi, inv);
-      x = inv * rs;
 
-      gamma = f1 -> HesseNorm() + f2 -> HesseNorm() + f3 -> HesseNorm();
-      beta = 0;
-      for (i = 0; i < 3; i++)
-	{
-	  sum = 0;
-	  for (j = 0; j < 3; j++)
-	    sum += fabs (inv(i,j));
-	  beta = max2 (beta, sum);
-	}
-      eta = Abs (x);
+  bool SpecialPointCalculation :: 
+  CrossPointNewtonConvergence (const Surface * f1, 
+			       const Surface * f2, 
+			       const Surface * f3,
+			       const Point<3> & p)
+  {
+    Vec<3> grad, rs, x;
+    Mat<3> jacobi, inv;
 
-      alpha = beta * gamma * eta;
-    }
+    f1->CalcGradient (p, grad);
+    jacobi(0,0) = grad(0);
+    jacobi(0,1) = grad(1);
+    jacobi(0,2) = grad(2);
 
-  return (alpha < 0.1);
-}
+    f2->CalcGradient (p, grad);
+    jacobi(1,0) = grad(0);
+    jacobi(1,1) = grad(1);
+    jacobi(1,2) = grad(2);
 
+    f3->CalcGradient (p, grad);
+    jacobi(2,0) = grad(0);
+    jacobi(2,1) = grad(1);
+    jacobi(2,2) = grad(2);
 
+    if (fabs (Det (jacobi)) > 1e-8)
+      {
+	double gamma = f1 -> HesseNorm() + f2 -> HesseNorm() + f3 -> HesseNorm();
+	if (gamma == 0.0) return 1;
 
+	CalcInverse (jacobi, inv);
 
-bool SpecialPointCalculation :: 
-CrossPointDegenerated (const Surface * f1,
-		       const Surface * f2, 
-		       const Surface * f3, 
-		       const BoxSphere<3> & box) const
-{
-  Mat<3> mat;
-  Vec<3> grad, g1, g2, g3;
-  double normprod;
+	rs(0) = f1->CalcFunctionValue (p);
+	rs(1) = f2->CalcFunctionValue (p);
+	rs(2) = f3->CalcFunctionValue (p);
 
-  if (box.Diam() > relydegtest) return 0;
+	x = inv * rs;
 
-  f1->CalcGradient (box.Center(), g1);
-  normprod = Abs (g1);
+	double beta = 0;
+	for (int i = 0; i < 3; i++)
+	  {
+	    double sum = 0;
+	    for (int j = 0; j < 3; j++)
+	      sum += fabs (inv(i,j));
+	    if (sum > beta)  beta = sum;
+	  }
+	double eta = Abs (x);
 
-  f2->CalcGradient (box.Center(), g2);
-  normprod *= Abs (g2);
- 
-  f3->CalcGradient (box.Center(), g3);
-  normprod *= Abs (g3);
+	return (beta * gamma * eta < 0.1);
+      }
+    return 0;
 
-  for (int i = 0; i < 3; i++)
-    {
-      mat(i,0) = g1(i);
-      mat(i,1) = g2(i);
-      mat(i,2) = g3(i);
-    }
+  }
 
-  if (fabs (Det (mat)) < cpeps1 * normprod)
-    return 1;
-  else 
-    return 0;
-}
- 
 
 
 
+  bool SpecialPointCalculation :: 
+  CrossPointDegenerated (const Surface * f1,
+			 const Surface * f2, 
+			 const Surface * f3, 
+			 const BoxSphere<3> & box) const
+  {
+    Mat<3> mat;
+    Vec<3> g1, g2, g3;
+    double normprod;
 
-void SpecialPointCalculation :: CrossPointNewton (const Surface * f1, 
-						  const Surface * f2, 
-						  const Surface * f3, Point<3> & p)
-{
-  int i;
-  Vec<3> g1, g2, g3;
-  Vec<3> rs, sol;
-  Mat<3> mat;
+    if (box.Diam() > relydegtest) return 0;
 
-  i = 10;
-  while (i > 0)
-    {
-      i--;
-      rs(0) = f1->CalcFunctionValue (p);
-      rs(1) = f2->CalcFunctionValue (p);
-      rs(2) = f3->CalcFunctionValue (p);
+    f1->CalcGradient (box.Center(), g1);
+    normprod = Abs2 (g1);
 
-      f1->CalcGradient (p, g1);
-      f2->CalcGradient (p, g2);
-      f3->CalcGradient (p, g3);
+    f2->CalcGradient (box.Center(), g2);
+    normprod *= Abs2 (g2);
+ 
+    f3->CalcGradient (box.Center(), g3);
+    normprod *= Abs2 (g3);
 
-      for (int j = 0; j < 3; j++)
-	{
-	  mat(0, j) = g1(j);
-	  mat(1, j) = g2(j);
-	  mat(2, j) = g3(j);
-	}
-      mat.Solve (rs, sol);
-	  /*
-      Transpose (g1, g2, g3);
-      SolveLinearSystem (g1, g2, g3, rs, sol);
-	  */
-      if (sol.Length() < 1e-12 && i > 1) i = 1;
-
-      p -= sol;
-    }
-}
+    for (int i = 0; i < 3; i++)
+      {
+	mat(i,0) = g1(i);
+	mat(i,1) = g2(i);
+	mat(i,2) = g3(i);
+      }
 
+    return sqr (Det (mat)) < sqr(cpeps1) * normprod;
+  }
+ 
 
 
 
-/******* Tests for Point on edges **********************/
 
+  void SpecialPointCalculation :: CrossPointNewton (const Surface * f1, 
+						    const Surface * f2, 
+						    const Surface * f3, Point<3> & p)
+  {
+    Vec<3> g1, g2, g3;
+    Vec<3> rs, sol;
+    Mat<3> mat;
 
+    int i = 10;
+    while (i > 0)
+      {
+	i--;
+	rs(0) = f1->CalcFunctionValue (p);
+	rs(1) = f2->CalcFunctionValue (p);
+	rs(2) = f3->CalcFunctionValue (p);
 
+	f1->CalcGradient (p, g1);
+	f2->CalcGradient (p, g2);
+	f3->CalcGradient (p, g3);
 
-bool SpecialPointCalculation :: 
-EdgeNewtonConvergence (const Surface * f1, const Surface * f2, 
-		       const Point<3> & p)
-{
-  int i;
-  Vec<3> g1, g2, sol;
-  Vec<2> vrs;
-  double alpha, beta, gamma, eta;
-  double sum;
-  Mat<2,3> mat;
-  Mat<3,2> inv;
-  int j;
-
-  vrs(0) = f1->CalcFunctionValue (p);
-  vrs(1) = f2->CalcFunctionValue (p);
-
-  f1->CalcGradient (p, g1);
-  f2->CalcGradient (p, g2);
-
-  for (i = 0; i < 3; i++)
-    {
-      mat(0,i) = g1(i);
-      mat(1,i) = g2(i);
-    }
+	for (int j = 0; j < 3; j++)
+	  {
+	    mat(0, j) = g1(j);
+	    mat(1, j) = g2(j);
+	    mat(2, j) = g3(j);
+	  }
+	mat.Solve (rs, sol);
+	if (sol.Length2() < 1e-24 && i > 1) i = 1;
 
-  alpha = 1;
+	p -= sol;
+      }
+  }
 
-  if ( fabs(g1 * g2) < (1 - 1e-8) * Abs (g1) * Abs (g2))
-    {
-      CalcInverse (mat, inv);
-      sol = inv * vrs;
 
-      // SolveLinearSystemLS (g1, g2, vrs, sol);
 
-      gamma = f1 -> HesseNorm() + f2 -> HesseNorm();
 
-      /*
-      Vec<3> inv1, inv2;
-      PseudoInverse (g1, g2, inv1, inv2);
-      */
-      
-      beta = 0;
-      for (i = 0; i < 3; i++)
-	for (j = 0; j < 2; j++)
-	  beta += inv(i,j) * inv(i,j);
-      beta = sqrt (beta);
-
-      //      beta = inv1.Length() + inv2.Length();
-      eta = Abs (sol);
-      alpha = beta * gamma * eta;
-    }
-  return (alpha < 0.1);
-}
+  /******* Tests for Point on edges **********************/
 
 
 
 
-bool SpecialPointCalculation :: 
-EdgeDegenerated (const Surface * f1,
-		 const Surface * f2, 
-		 const BoxSphere<3> & box) const
-{
-  // perform newton steps. normals parallel ?
-  // if not decideable: return 0 
-  
-  
-  Point<3> p = box.Center();
-  int i;
-  Vec<3> grad, g1, g2, sol;
-  Vec<2> vrs;
-  Mat<2,3> mat;
-
-  i = 20;
-  while (i > 0)
-    {
-      if (Dist (p, box.Center()) > box.Diam())
-	return 0;
+  bool SpecialPointCalculation :: 
+  EdgeNewtonConvergence (const Surface * f1, const Surface * f2, 
+			 const Point<3> & p)
+  {
+    Vec<3> g1, g2, sol;
+    Vec<2> vrs;
+    Mat<2,3> mat;
+    Mat<3,2> inv;
 
-      i--;
-      vrs(0) = f1->CalcFunctionValue (p);
-      vrs(1) = f2->CalcFunctionValue (p);
+    f1->CalcGradient (p, g1);
+    f2->CalcGradient (p, g2);
 
-      f1->CalcGradient (p, g1);
-      f2->CalcGradient (p, g2);
+    if ( sqr(g1 * g2) < (1 - 1e-8) * Abs2 (g1) * Abs2 (g2))
+      {
+	double gamma = f1 -> HesseNorm() + f2 -> HesseNorm();
+	if (gamma < 1e-32) return 1;
+	gamma = sqr (gamma);
+      
+	for (int i = 0; i < 3; i++)
+	  {
+	    mat(0,i) = g1(i);
+	    mat(1,i) = g2(i);
+	  }
 
-      if ( fabs (g1 * g2) > (1 - 1e-10) * Abs (g1) * Abs (g2))
-	return 1;
+	CalcInverse (mat, inv);
 
-      for (int j = 0; j < 3; j++)
-	{
-	  mat(0,j) = g1(j);
-	  mat(1,j) = g2(j);
-	}
-      mat.Solve (vrs, sol);
-      //      SolveLinearSystemLS (g1, g2, vrs, sol);
+	vrs(0) = f1->CalcFunctionValue (p);
+	vrs(1) = f2->CalcFunctionValue (p);
+	sol = inv * vrs;
 
-      if (Abs (sol) < 1e-12 && i > 1) i = 1;
-      p -= sol;
-    }
+	double beta = 0;
+	for (int i = 0; i < 3; i++)
+	  for (int j = 0; j < 2; j++)
+	    beta += inv(i,j) * inv(i,j);
+	// beta = sqrt (beta);
 
-  return 0;
-  /*
-  return 0;
+	double eta = Abs2 (sol);
 
-  static DenseMatrix jacobi(3);
-  Vec<3> grad, g1, g2, g3;
-  double normprod;
-  
-  if (box.Diam() > relydegtest) return 0;
-  
-  f1->CalcGradient (box.Center(), g1);
-  normprod = g1.Length();
-  
-  f2->CalcGradient (box.Center(), g2);
-  normprod *= g2.Length();
-  
-  if (fabs (g1 * g2) < 1e-8 * normprod)
-    return 1;
-  else 
+	// alpha = beta * gamma * eta;
+	return (beta * gamma * eta < 0.01);
+      }
     return 0;
-  */
-}
+  }
 
 
 
 
+  bool SpecialPointCalculation :: 
+  EdgeDegenerated (const Surface * f1,
+		   const Surface * f2, 
+		   const BoxSphere<3> & box) const
+  {
+    // perform newton steps. normals parallel ?
+    // if not decideable: return 0 
+  
+    Point<3> p = box.Center();
+    Vec<3> g1, g2, sol;
+    Vec<2> vrs;
+    Mat<2,3> mat;
 
+    int i = 20;
+    while (i > 0)
+      {
+	if (Dist2 (p, box.Center()) > sqr(box.Diam()))
+	  return 0;
 
-void SpecialPointCalculation :: EdgeNewton (const Surface * f1, 
-					    const Surface * f2, Point<3> & p)
-{
-  int i;
-  Vec<3> grad, g1, g2, sol;
-  Vec<2> vrs;
-  Mat<2,3> mat;
+	i--;
+	vrs(0) = f1->CalcFunctionValue (p);
+	vrs(1) = f2->CalcFunctionValue (p);
 
-  i = 10;
-  while (i > 0)
-    {
-      i--;
-      vrs(0) = f1->CalcFunctionValue (p);
-      vrs(1) = f2->CalcFunctionValue (p);
+	f1->CalcGradient (p, g1);
+	f2->CalcGradient (p, g2);
 
-      f1->CalcGradient (p, g1);
-      f2->CalcGradient (p, g2);
+	if ( sqr (g1 * g2) > (1 - 1e-10) * Abs2 (g1) * Abs2 (g2))
+	  return 1;
 
-      for (int j = 0; j < 3; j++)
-	{
-	  mat(0,j) = g1(j);
-	  mat(1,j) = g2(j);
-	}
-      mat.Solve (vrs, sol);
-      //      SolveLinearSystemLS (g1, g2, vrs, sol);
+	for (int j = 0; j < 3; j++)
+	  {
+	    mat(0,j) = g1(j);
+	    mat(1,j) = g2(j);
+	  }
+	mat.Solve (vrs, sol);
 
-      if (Abs (sol) < 1e-12 && i > 1) i = 1;
-      p -= sol;
-    }
-}
+	if (Abs2 (sol) < 1e-24 && i > 1) i = 1;
+	p -= sol;
+      }
 
+    return 0;
+  }
 
 
-bool SpecialPointCalculation :: 
-IsEdgeExtremalPoint (const Surface * f1, const Surface * f2, 
-		     const Point<3> & p, Point<3> & pp, double rad)
-{
-  Vec<3> g1, g2, t, t1, t2;
-  int j;
 
-  f1->CalcGradient (p, g1);
-  f2->CalcGradient (p, g2);
-  
-  t = Cross (g1, g2);
-  t.Normalize();
-
-  Point<3> p1 = p + rad * t;
-  Point<3> p2 = p - rad * t;
 
-  EdgeNewton (f1, f2, p1);
-  EdgeNewton (f1, f2, p2);
 
-  
-  f1->CalcGradient (p1, g1);
-  f2->CalcGradient (p1, g2);
-  t1 = Cross (g1, g2);
-  t1.Normalize();
-
-  f1->CalcGradient (p2, g1);
-  f2->CalcGradient (p2, g2);
-  t2 = Cross (g1, g2);
-  t2.Normalize();
-
-  double val = 1e-8 * rad * rad;
-  for (j = 0; j < 3; j++)
-    if ( (t1(j) * t2(j) < -val) )
-      {
-	pp = p;
-	ExtremalPointNewton (f1, f2, j+1, pp);
-	return 1;
-      }
 
-  return 0;
-}
+  void SpecialPointCalculation :: EdgeNewton (const Surface * f1, 
+					      const Surface * f2, Point<3> & p)
+  {
+    Vec<3> g1, g2, sol;
+    Vec<2> vrs;
+    Mat<2,3> mat;
 
+    int i = 10;
+    while (i > 0)
+      {
+	i--;
+	vrs(0) = f1->CalcFunctionValue (p);
+	vrs(1) = f2->CalcFunctionValue (p);
 
+	f1->CalcGradient (p, g1);
+	f2->CalcGradient (p, g2);
 
+	for (int j = 0; j < 3; j++)
+	  {
+	    mat(0,j) = g1(j);
+	    mat(1,j) = g2(j);
+	  }
+	mat.Solve (vrs, sol);
 
+	if (Abs2 (sol) < 1e-24 && i > 1) i = 1;
+	p -= sol;
+      }
+  }
 
 
 
+  bool SpecialPointCalculation :: 
+  IsEdgeExtremalPoint (const Surface * f1, const Surface * f2, 
+		       const Point<3> & p, Point<3> & pp, double rad)
+  {
+    Vec<3> g1, g2, t, t1, t2;
 
+    f1->CalcGradient (p, g1);
+    f2->CalcGradient (p, g2);
+  
+    t = Cross (g1, g2);
+    t.Normalize();
 
-/********** Tests of Points of extremal coordinates  ****************/
+    Point<3> p1 = p + rad * t;
+    Point<3> p2 = p - rad * t;
 
+    EdgeNewton (f1, f2, p1);
+    EdgeNewton (f1, f2, p2);
+  
+    f1->CalcGradient (p1, g1);
+    f2->CalcGradient (p1, g2);
+    t1 = Cross (g1, g2);
+    t1.Normalize();
+
+    f1->CalcGradient (p2, g1);
+    f2->CalcGradient (p2, g2);
+    t2 = Cross (g1, g2);
+    t2.Normalize();
+
+    double val = 1e-8 * rad * rad;
+    for (int j = 0; j < 3; j++)
+      if ( (t1(j) * t2(j) < -val) )
+	{
+	  pp = p;
+	  ExtremalPointNewton (f1, f2, j+1, pp);
+	  return 1;
+	}
 
-void SpecialPointCalculation :: ExtremalPointNewton (const Surface * f1, 
-						     const Surface * f2, 
-						     int dir, Point<3> & p)
-{
-  int i;
+    return 0;
+  }
 
-  Vec<3> g1, g2, v, curv;
-  Vec<3> rs, x, y1, y2, y;
-  Mat<3> h1, h2;
-  Mat<3> jacobi;
 
 
-  if (dir < 1 || dir > 3)
-    {
-      cerr << "Error: Illegal extremdir" << endl;
-      return;
-    }
 
-  i = 50;
-  while (i > 0)
-    {
-      i--;
-      rs(0) = f1->CalcFunctionValue (p);
-      rs(1) = f2->CalcFunctionValue (p);
 
-      f1 -> CalcGradient (p, g1);
-      f2 -> CalcGradient (p, g2);
 
-      f1 -> CalcHesse (p, h1);
-      f2 -> CalcHesse (p, h2);
 
 
-      v = Cross (g1, g2);
 
-      rs(2) = v(dir-1);
+  /********** Tests of Points of extremal coordinates  ****************/
 
-      jacobi(0,0) = g1(0);
-      jacobi(0,1) = g1(1);
-      jacobi(0,2) = g1(2);
 
-      jacobi(1,0) = g2(0);
-      jacobi(1,1) = g2(1);
-      jacobi(1,2) = g2(2);
+  void SpecialPointCalculation :: ExtremalPointNewton (const Surface * f1, 
+						       const Surface * f2, 
+						       int dir, Point<3> & p)
+  {
+    Vec<3> g1, g2, v, curv;
+    Vec<3> rs, x, y1, y2, y;
+    Mat<3> h1, h2;
+    Mat<3> jacobi;
+
+    int i = 50;
+    while (i > 0)
+      {
+	i--;
+	rs(0) = f1->CalcFunctionValue (p);
+	rs(1) = f2->CalcFunctionValue (p);
 
+	f1 -> CalcGradient (p, g1);
+	f2 -> CalcGradient (p, g2);
 
-      switch (dir)
-	{
-	case 1:
-	  {
-	    y1(0) = 0;
-	    y1(1) = g2(2);
-	    y1(2) = -g2(1);
-	    y2(0) = 0;
-	    y2(1) = -g1(2);
-	    y2(2) = g1(1);
-	    break;
-	  }
-	case 2:
-	  {
-	    y1(0) = -g2(2);
-	    y1(1) = 0;
-	    y1(2) = g2(0);
-	    y2(0) = g1(2);
-	    y2(1) = 0;
-	    y2(2) = -g1(0);
-	    break;
-	  }
-	case 3:
-	  {
-	    y1(0) = g2(1);
-	    y1(1) = -g2(0);
-	    y1(2) = 0;
-	    y2(0) = -g1(1);
-	    y2(1) = g1(0);
-	    y2(2) = 0;
-	    break;
-	  }
-	}
+	f1 -> CalcHesse (p, h1);
+	f2 -> CalcHesse (p, h2);
 
-      y = h1 * y1 + h2 * y2;
+	v = Cross (g1, g2);
 
-      jacobi(2,0) = y(0);
-      jacobi(2,1) = y(1);
-      jacobi(2,2) = y(2);
+	rs(2) = v(dir-1);
 
-      jacobi.Solve (rs, x);
-      /*
-      CalcInverse (jacobi, inv);
-      inv.Mult (rs, x);
-      */
-      //    (*testout) << "err = " << x.L2Norm() << endl;
+	jacobi(0,0) = g1(0);
+	jacobi(0,1) = g1(1);
+	jacobi(0,2) = g1(2);
 
-      if (Abs (x) < 1e-12 && i > 1)
-	{
-	  //      (*testout) << "convergent in " << (10 - i) << " steps " << endl;
+	jacobi(1,0) = g2(0);
+	jacobi(1,1) = g2(1);
+	jacobi(1,2) = g2(2);
 
-	  i = 1;
-	}
-      
-      p -= x;
-    }
 
-  if (Abs (x) > 1e-10)
-    {
-      (*testout) << "Error: extremum Newton not convergent" << endl;
-      (*testout) << "dir = " << dir << endl;
-      (*testout) << "p = " << p << endl;
-      (*testout) << "x = " << x << endl;
-    }
-}
+	switch (dir)
+	  {
+	  case 1:
+	    {
+	      y1(0) = 0;
+	      y1(1) = g2(2);
+	      y1(2) = -g2(1);
+	      y2(0) = 0;
+	      y2(1) = -g1(2);
+	      y2(2) = g1(1);
+	      break;
+	    }
+	  case 2:
+	    {
+	      y1(0) = -g2(2);
+	      y1(1) = 0;
+	      y1(2) = g2(0);
+	      y2(0) = g1(2);
+	      y2(1) = 0;
+	      y2(2) = -g1(0);
+	      break;
+	    }
+	  case 3:
+	    {
+	      y1(0) = g2(1);
+	      y1(1) = -g2(0);
+	      y1(2) = 0;
+	      y2(0) = -g1(1);
+	      y2(1) = g1(0);
+	      y2(2) = 0;
+	      break;
+	    }
+	  }
 
+	y = h1 * y1 + h2 * y2;
 
+	jacobi(2,0) = y(0);
+	jacobi(2,1) = y(1);
+	jacobi(2,2) = y(2);
 
+	jacobi.Solve (rs, x);
 
-bool SpecialPointCalculation :: ExtremalPointPossible (const Surface * f1, 
-						       const Surface * f2, 
-						       int dir, 
-						       const BoxSphere<3> & box)
-{
-  double hn1, hn2, gn1, gn2;
-  Point<3> p;
-  Vec<3> g1, g2, v;
-  double f3;
-  double r = box.Diam()/2;
+	if (Abs2 (x) < 1e-24 && i > 1)
+	  {
+	    i = 1;
+	  }
+      
+	p -= x;
+      }
 
-  p = box.Center();
 
-  f1 -> CalcGradient (p, g1);
-  f2 -> CalcGradient (p, g2);
+    if (Abs2 (x) > 1e-20)
+      {
+	(*testout) << "Error: extremum Newton not convergent" << endl;
+	(*testout) << "dir = " << dir << endl;
+	(*testout) << "p = " << p << endl;
+	(*testout) << "x = " << x << endl;
+      }
+  }
 
-  gn1 = g1.Length();
-  gn2 = g2.Length();
 
-  hn1 = f1 -> HesseNorm ();
-  hn2 = f2 -> HesseNorm ();
 
-  v = Cross (g1, g2);
-  f3 = fabs (v(dir-1));
+  void SpecialPointCalculation :: 
+  ComputeExtremalPoints (const Plane * plane, 
+			 const QuadraticSurface * quadric, 
+			 ARRAY<Point<3> > & pts)
+  {
+    // 3 equations:
+    // surf1 = 0  <===> plane_a + plane_b x = 0;
+    // surf2 = 0  <===> quad_a + quad_b x + x^T quad_c x = 0
+    // (grad 1 x grad 2)(i) = 0  <====> (grad 1 x e_i) . grad_2 = 0
 
-  //  (*testout) << "f3 = " << f3 << "  r = " << r 
-  //             << "normbound = " 
-  //             << (hn1 * (gn2 + r * hn2) + hn2 * (gn1 + r * hn1)) << endl;
- 
-  return (f3 <= 3 * r * (hn1 * (gn2 + r * hn2) + hn2 * (gn1 + r * hn1)));
-}
+    pts.SetSize (0);
 
+    Point<3> p0(0,0,0);
+    double plane_a, quad_a;
+    Vec<3> plane_b, quad_b, ei;
+    Mat<3> quad_c;
 
+    plane_a = plane -> CalcFunctionValue(p0);
+    plane -> CalcGradient (p0, plane_b);
 
-bool SpecialPointCalculation :: 
-ExtremalPointNewtonConvergence (const Surface * f1, const Surface * f2, 
-				int dir, 
-				const BoxSphere<3> & box)
-{
-  return box.Diam() < 1e-8;
-}
+    quad_a = quadric -> CalcFunctionValue(p0);
+    quadric -> CalcGradient (p0, quad_b);
+    quadric -> CalcHesse (p0, quad_c);
+    for (int i = 0; i < 3; i++)
+      for (int j = 0; j < 3; j++)
+	quad_c(i,j) *= 0.5;
 
+    for (int dir = 0; dir <= 2; dir++)
+      {
+	ei = 0.0; ei(dir) = 1;
+	Vec<3> v1 = Cross (plane_b, ei);
+	
+	// grad_2 . v1 ... linear:
+	double g2v1_c = v1 * quad_b;
+	Vec<3> g2v1_l = 2.0 * (quad_c * v1);
+
+	// find line of two linear equations:
+	
+	Vec<2> rhs;
+	Vec<3> sol;
+	Mat<2,3> mat;
+
+	for (int j = 0; j < 3; j++)
+	  {
+	    mat(0,j) = plane_b(j);
+	    mat(1,j) = g2v1_l(j);
+	  }
+	rhs(0) = -plane_a;
+	rhs(1) = -g2v1_c;
 
-bool SpecialPointCalculation :: 
-ExtremalPointDegenerated (const Surface * f1, const Surface * f2, 
-			  int dir, const BoxSphere<3> & box)
-{
-  double gn1, gn2;
-  Point<3> p;
-  Vec<3> g1, g2, v;
-  double maxderiv;
-  double minv;
-  Vec<3> curv, t;
-  Vec<2> rs, x;
-  Mat<3> h1, h2;
-  Mat<2> a, inv;
-  double leftside;
-
-  if (box.Diam() > relydegtest) return 0;
-
-  p = box.Center();
-
-  f1 -> CalcGradient (p, g1);
-  f2 -> CalcGradient (p, g2);
-  gn1 = g1.Length();
-  gn2 = g2.Length();
-
-  v = Cross (g1, g2);
-  if (Abs (v) < epeps1 * gn1 * gn2) return 1;       // irregular edge
-
-  f1 -> CalcHesse (p, h1);
-  f2 -> CalcHesse (p, h2);
-
-  //  hn1 = f1 -> HesseNorm ();
-  //  hn2 = f2 -> HesseNorm ();
-
-  t = v;
-  a(0, 0) = g1 * g1;
-  a(0, 1) = 
-    a(1, 0) = g1 * g2;
-  a(1, 1) = g2 * g2;
-  
-  rs(0) = g1(dir-1);
-  rs(1) = g2(dir-1);
+	Vec<3> t = Cross (plane_b, g2v1_l);
+	if (Abs2(t) > 0)
+	  {
+	    mat.Solve (rhs, sol);
+	    
+	    // solve quadratic equation along line  sol + alpha t ....
+	    double a = quad_a + quad_b * sol + sol * (quad_c * sol);
+	    double b = quad_b * t + 2 * (sol * (quad_c * t));
+	    double c = t * (quad_c * t);
 
-  a.Solve (rs, x);
+	    // solve a + b alpha + c alpha^2:
 
-  /*
-  CalcInverse (a, inv);
-  inv.Mult (rs, x);          // x .. Lagrangeparameter
-  */
-  //  (*testout) << "g1 = " << g1 << " g2 = " << g2 << endl;
-  //  (*testout) << "lam = " << x << endl;
-  //  (*testout) << "h2 = " << h2 << endl;
+	    if (fabs (c) > 1e-32)
+	      {
+		double disc = sqr (0.5*b/c) - a/c;
+		if (disc > 0)
+		  {
+		    disc = sqrt (disc);
+		    double alpha1 = -0.5*b/c + disc;
+		    double alpha2 = -0.5*b/c - disc;
+
+		    pts.Append (Point<3> (sol+alpha1*t));
+		    pts.Append (Point<3> (sol+alpha2*t));
+		    /*
+		    cout << "sol1 = " << sol + alpha1 * t
+			 << ", sol2 = " << sol + alpha2 * t << endl;
+		    */
+		  }
+	      }
+	  }
+      }
+  }
 
-  leftside = fabs (x(0) * ( t * (h1 * t)) + 
-                   x(1) * ( t * (h2 * t)));
 
-  //  (*testout) << "leftside = " << leftside << endl;
 
-  if (leftside < epeps2 * Abs2 (v)) return 1;  
 
-  return 0;
-}
 
- 
 
-bool SpecialPointCalculation :: AddPoint (const Point<3> & p, int layer)
-{
-  for (int i = 0; i < points->Size(); i++)
-    if (Dist2 ( (*points)[i], p) < epspointdist2 &&
-	(*points)[i].GetLayer() == layer)
-      return 0;
+  /*
+    bool SpecialPointCalculation :: ExtremalPointPossible (const Surface * f1, 
+    const Surface * f2, 
+    int dir, 
+    const BoxSphere<3> & box)
+    {
+    double hn1, hn2, gn1, gn2;
+    Point<3> p;
+    Vec<3> g1, g2, v;
+    double f3;
+    double r = box.Diam()/2;
 
-  points->Append (MeshPoint(p, layer));
-  return 1;
-}
+    p = box.Center();
 
+    f1 -> CalcGradient (p, g1);
+    f2 -> CalcGradient (p, g2);
 
+    gn1 = g1.Length();
+    gn2 = g2.Length();
 
+    hn1 = f1 -> HesseNorm ();
+    hn2 = f2 -> HesseNorm ();
 
+    v = Cross (g1, g2);
+    f3 = fabs (v(dir-1));
 
+    //  (*testout) << "f3 = " << f3 << "  r = " << r 
+    //             << "normbound = " 
+    //             << (hn1 * (gn2 + r * hn2) + hn2 * (gn1 + r * hn1)) << endl;
+ 
+    return (f3 <= 3 * r * (hn1 * (gn2 + r * hn2) + hn2 * (gn1 + r * hn1)));
+    }
 
 
-/*
-void SpecialPointCalculation :: 
-AnalyzeSpecialPoints (const CSGeometry & ageometry,
-		      ARRAY<Point<3> > & apoints, 
-		      ARRAY<SpecialPoint> & specpoints)
-{
-  int si, i, j, k, l, m, spi;
-  Solid * locsol;
-  ARRAY<int> surfind;
-  ARRAY<Vec<3>> normalvecs;
-  const Solid * sol;
-  Vec<3> t;
-  Point<3> p;
-
-  ARRAY<int> specpoint2point;
-  specpoints.SetSize (0);
- 
-  (*testout) << "AnalyzeSpecialPoints" << endl;
 
-  for (si = 1; si <= ageometry.GetNTopLevelObjects(); si++)
+    bool SpecialPointCalculation :: 
+    ExtremalPointNewtonConvergence (const Surface * f1, const Surface * f2, 
+    int dir, 
+    const BoxSphere<3> & box)
     {
-      (*testout) << "main solid " << si << endl;
-
-      sol = ageometry.GetTopLevelObject(si)->GetSolid();
-      for (i = 1; i <= apoints.Size(); i++)
-	{
-	  p = apoints.Get(i);
+    return box.Diam() < 1e-8;
+    }
 
-	  sol -> TangentialSolid (p, locsol);
-	  if (!locsol) continue;
-	  
-	  (*testout) << "Point " << apoints.Get(i) << endl;
 
-	  locsol -> GetSurfaceIndices (surfind);
-	  for (j = surfind.Size(); j >= 1; j--)
-	    if (fabs (ageometry.GetSurface(surfind.Get(j))->
-		      CalcFunctionValue (p)) > 1e-6)
-	      surfind.DeleteElement (j);
+    bool SpecialPointCalculation :: 
+    ExtremalPointDegenerated (const Surface * f1, const Surface * f2, 
+    int dir, const BoxSphere<3> & box)
+    {
+    double gn1, gn2;
+    Point<3> p;
+    Vec<3> g1, g2, v;
+    double maxderiv;
+    double minv;
+    Vec<3> curv, t;
+    Vec<2> rs, x;
+    Mat<3> h1, h2;
+    Mat<2> a, inv;
+    double leftside;
+
+    if (box.Diam() > relydegtest) return 0;
+
+    p = box.Center();
+
+    f1 -> CalcGradient (p, g1);
+    f2 -> CalcGradient (p, g2);
+    gn1 = g1.Length();
+    gn2 = g2.Length();
+
+    v = Cross (g1, g2);
+    if (Abs (v) < epeps1 * gn1 * gn2) return 1;       // irregular edge
+
+    f1 -> CalcHesse (p, h1);
+    f2 -> CalcHesse (p, h2);
+
+    //  hn1 = f1 -> HesseNorm ();
+    //  hn2 = f2 -> HesseNorm ();
+
+    t = v;
+    a(0, 0) = g1 * g1;
+    a(0, 1) = 
+    a(1, 0) = g1 * g2;
+    a(1, 1) = g2 * g2;
+  
+    rs(0) = g1(dir-1);
+    rs(1) = g2(dir-1);
 
+    a.Solve (rs, x);
 
+    //  (*testout) << "g1 = " << g1 << " g2 = " << g2 << endl;
+    //  (*testout) << "lam = " << x << endl;
+    //  (*testout) << "h2 = " << h2 << endl;
 
-	  (*testout) << "Surfaces: ";
-	  for (j = 1; j <= surfind.Size(); j++)
-	    (*testout) << surfind.Get(j) << " ";
-	  (*testout) << endl;
+    leftside = fabs (x(0) * ( t * (h1 * t)) + 
+    x(1) * ( t * (h2 * t)));
 
-	  normalvecs.SetSize(surfind.Size());
-	  for (j = 1; j <= surfind.Size(); j++)
-	    ageometry.GetSurface(surfind.Get(j)) -> 
-	      GetNormalVector(apoints.Get(i), normalvecs.Elem(j));
+    //  (*testout) << "leftside = " << leftside << endl;
 
-	  for (j = 1; j <= normalvecs.Size() - 1; j ++)
-	    for (k = j+1; k <= normalvecs.Size(); k++)
-	      for (l = 1; l <= 2; l++)
-		{
-		  t = Cross (normalvecs.Get(j), normalvecs.Get(k));
-		  if (t.Length2() < 1e-8)
-		    {
-		      cerr << "AnalyzePoint: Surfaces degenerated" << endl;
-		      break;
-		    }
-		  t /= t.Length();
-		  if (l == 2) t *= -1;
+    if (leftside < epeps2 * Abs2 (v)) return 1;  
 
-		  if (locsol -> Edge (apoints.Get(i), t))
-		    {
-		      spi = 0;
-		      for (m = 1; m <= specpoints.Size(); m++)
-			if (Dist2 (specpoints.Get(m).p, apoints.Get(i)) < 1e-8
-			    && (specpoints.Get(m).v - t).Length2() < 1e-8)
-			  {
-			    spi = m;
-			    break;
-			  }
-		      if (!spi)
-			{
-			  spi = specpoints.Append (SpecialPoint());
-			  specpoint2point.Append (i);
-			  specpoints.Last().unconditional = 0;
-			}
-		      specpoints.Elem(spi).p = apoints.Get(i);
-		      specpoints.Elem(spi).v = t;
-		      if (surfind.Size() >= 3)
-			specpoints.Elem(spi).unconditional = 1;
-		      specpoints.Elem(spi).s1 = surfind.Get(j);
-		      specpoints.Elem(spi).s2 = surfind.Get(k);
-		      (*testout) << "spi = " << spi 
-				 << " uncond = " << specpoints.Get(spi).unconditional
-				 << " t = " << t << endl;
-		    }
-          
-		}
-	  delete locsol;
-	}
+    return 0;
     }
+  */
+ 
 
-  // if special point is unconditional on some solid,
-  // it must be unconditional everywhere:
+  bool SpecialPointCalculation :: AddPoint (const Point<3> & p, int layer)
+  {
+    for (int i = 0; i < points->Size(); i++)
+      if (Dist2 ( (*points)[i], p) < epspointdist2 &&
+	  (*points)[i].GetLayer() == layer)
+	return 0;
 
-  BitArray uncond (apoints.Size());
-  uncond.Clear();
+    points->Append (MeshPoint(p, layer));
+    PrintMessageCR (3, "Found points ", points->Size());
+    return 1;
+  }
 
-  for (i = 1; i <= specpoints.Size(); i++)
-    if (specpoints.Get(i).unconditional)
-      uncond.Set (specpoint2point.Get(i));
 
-  for (i = 1; i <= specpoints.Size(); i++)
-    specpoints.Elem(i).unconditional = 
-      uncond.Test (specpoint2point.Get(i)) ? 1 : 0;
-}
-*/
 
 
 
-void SpecialPointCalculation :: 
-AnalyzeSpecialPoints (const CSGeometry & ageometry,
-		      ARRAY<MeshPoint> & apoints, 
-		      ARRAY<SpecialPoint> & specpoints)
-{
-  int si, i, j, k, l, m, spi;
 
-  Solid * locsol;
-  ARRAY<int> surfind;
-  ARRAY<int> surfind2;
 
-  ARRAY<Vec<3> > normalvecs;
-  const Solid * sol;
-  const Surface * surf;
+  void SpecialPointCalculation :: 
+  AnalyzeSpecialPoints (const CSGeometry & ageometry,
+			ARRAY<MeshPoint> & apoints, 
+			ARRAY<SpecialPoint> & specpoints)
+  {
+    ARRAY<int> surfind;
+    ARRAY<int> surfind2;
 
-  Vec<3> t, nsurf;
-  Point<3> p;
+    ARRAY<Vec<3> > normalvecs;
+    Vec<3> t, nsurf;
+    Point<3> p;
 
-  ARRAY<int> specpoint2point;
-  specpoints.SetSize (0);
+    ARRAY<int> specpoint2point;
+    specpoints.SetSize (0);
 
-  geometry = &ageometry;
+    geometry = &ageometry;
  
-  (*testout) << "AnalyzeSpecialPoints\n";
-
-  for (si = 0; si < ageometry.GetNTopLevelObjects(); si++)
-    {
-      (*testout) << "main solid " << si << "\n";
+    (*testout) << "AnalyzeSpecialPoints\n";
 
-      sol = ageometry.GetTopLevelObject(si)->GetSolid();
-      surf = ageometry.GetTopLevelObject(si)->GetSurface();
 
-      for (i = 0; i < apoints.Size(); i++)
-	{
-	  p = apoints[i];
-	  if (ageometry.GetTopLevelObject(si)->GetLayer() !=
-	      apoints[i].GetLayer())
-	    continue;
-
-	  // (*testout) << "Point " << apoints[i] << "\n";
-
-	  sol -> TangentialSolid (p, locsol);
-	  if (!locsol) continue;
-	  
-	  // get all surface indices, 
-	  if (surf)
-	    {
-	      locsol -> GetSurfaceIndices (surfind);
-	      bool hassurf = 0;
-	      for (m = 0; m < surfind.Size(); m++)
-		if (ageometry.GetSurface(surfind[m]) == surf)
-		  hassurf = 1;
-
-	      if (!hassurf)
-		continue;
+    Box<3> bbox;
+    if (apoints.Size())
+      bbox.Set (apoints[0]);
+    else
+      { bbox.Set (Point<3> (0,0,0)); bbox.Add (Point<3> (1,1,1)); }
+    for (int i = 1; i < apoints.Size(); i++)
+      bbox.Add (apoints[i]);
+    bbox.Increase (0.1 * Dist (bbox.PMin(), bbox.PMax()));
 
-	      surf->GetNormalVector (p, nsurf);
-	    }
-
-	  // get independent surfaces of tangential solid
-
-	  BoxSphere<3> box(p,p);
-	  box.Increase (1e-6);
-	  box.CalcDiamCenter();
-	  ageometry.GetIndependentSurfaceIndices (locsol, box, surfind);
+    Point3dTree searchtree (bbox.PMin(), bbox.PMax());
+    ARRAY<int> locsearch;
 
+    for (int si = 0; si < ageometry.GetNTopLevelObjects(); si++)
+      {
+	//      (*testout) << "main solid " << si << "\n";
+	
+	const Solid * sol = ageometry.GetTopLevelObject(si)->GetSolid();
+	const Surface * surf = ageometry.GetTopLevelObject(si)->GetSurface();
 
-	  /*
-	  locsol -> GetSurfaceIndices (surfind);
-	  for (j = surfind.Size(); j >= 1; j--)
-	    if (fabs (ageometry.GetSurface(surfind.Get(j))->
-		      CalcFunctionValue (p)) > 1e-6)
-	      surfind.DeleteElement (j);
-	  */
+	for (int i = 0; i < apoints.Size(); i++)
+	  {
+	    p = apoints[i];
+	    if (ageometry.GetTopLevelObject(si)->GetLayer() !=
+		apoints[i].GetLayer())
+	      continue;
 
-	  /*
-	  (*testout) << "Surfaces: ";
-	  for (j = 0; j < surfind.Size(); j++)
-	    (*testout) << surfind[j] << " ";
-	  (*testout) << "\n";
-	  */
+	    //	  (*testout) << "Point " << apoints[i] << "\n";
 
+	    Solid * locsol;
+	    sol -> TangentialSolid (p, locsol);
+	    if (!locsol) continue;
+	  
+	    // get all surface indices, 
+	    if (surf)
+	      {
+		locsol -> GetSurfaceIndices (surfind);
+		bool hassurf = 0;
+		for (int m = 0; m < surfind.Size(); m++)
+		  if (ageometry.GetSurface(surfind[m]) == surf)
+		    hassurf = 1;
 
-	  normalvecs.SetSize(surfind.Size());
-	  for (j = 0; j < surfind.Size(); j++)
-	    ageometry.GetSurface(surfind[j]) ->
-	      GetNormalVector(apoints[i], normalvecs[j]);
+		if (!hassurf)
+		  continue;
 
-	  for (j = 0; j < normalvecs.Size(); j++)
-	    for (k = j+1; k < normalvecs.Size(); k++)
-	      for (l = 1; l <= 2; l++)
-		{
-		  t = Cross (normalvecs[j], normalvecs[k]);
-		  if (Abs2 (t) < 1e-8)
-		    {
-		      cerr << "AnalyzePoint: Surfaces degenerated" << "\n";
-		      break;
-		    }
-		  t.Normalize();
-		  if (l == 2) t *= -1;
+		nsurf = surf->GetNormalVector (p);
+	      }
 
-		  // try tangential direction t
+	    // get independent surfaces of tangential solid
 
-		  // (*testout) << "check tangential " << t << "\n";
+	    BoxSphere<3> box(p,p);
+	    box.Increase (1e-6);
+	    box.CalcDiamCenter();
+	    ageometry.GetIndependentSurfaceIndices (locsol, box, surfind);
 
-		  if (surf && fabs (nsurf * t) > 1e-6)
-		    continue;
+	    normalvecs.SetSize(surfind.Size());
+	    for (int j = 0; j < surfind.Size(); j++)
+	      normalvecs[j] = 
+		ageometry.GetSurface(surfind[j]) -> GetNormalVector(apoints[i]);
 
-		  if (!surf)
-		    {
-		      ageometry.GetIndependentSurfaceIndices 
-			(locsol, p, t, surfind2);
+	    for (int j = 0; j < normalvecs.Size(); j++)
+	      for (int k = j+1; k < normalvecs.Size(); k++)
+		for (int l = 1; l <= 2; l++)
+		  {
+		    t = Cross (normalvecs[j], normalvecs[k]);
+		    if (Abs2 (t) < 1e-8)
+		      {
+			cerr << "AnalyzePoint: Surfaces degenerated" << "\n";
+			break;
+		      }
+		    t.Normalize();
+		    if (l == 2) t *= -1;
+
+		    // try tangential direction t
+
+		    // (*testout) << "check tangential " << t << "\n";
+
+		    if (surf && fabs (nsurf * t) > 1e-6)
+		      continue;
+
+		    if (!surf)
+		      {
+			ageometry.GetIndependentSurfaceIndices 
+			  (locsol, p, t, surfind2);
 		  
-		      bool found1 = 0, found2 = 0;
-		      for (int ii = 0; ii < surfind2.Size(); ii++)
-			{
-			  if (surfind2[ii] == surfind[j])
-			    found1 = 1;
-			  if (surfind2[ii] == surfind[k])
-			    found2 = 1;
-			}
-		      if (!found1 || !found2)
-			continue;
-		    }
+			bool found1 = 0, found2 = 0;
+			for (int ii = 0; ii < surfind2.Size(); ii++)
+			  {
+			    if (surfind2[ii] == surfind[j])
+			      found1 = 1;
+			    if (surfind2[ii] == surfind[k])
+			      found2 = 1;
+			  }
+			if (!found1 || !found2)
+			  continue;
+		      }
 
 
-		  bool isedge;
+		    bool isedge;
 
-		  // isedge = locsol -> Edge (apoints.Get(i), t);
+		    // isedge = locsol -> Edge (apoints.Get(i), t);
 		  
-		  // edge must be on tangential surface
-		  isedge = 
-		    locsol->VectorIn (p, t) &&
-		    !locsol->VectorStrictIn (p, t);
+		    // edge must be on tangential surface
+		    isedge = 
+		      locsol->VectorIn (p, t) &&
+		      !locsol->VectorStrictIn (p, t);
 		  
-		  // (*testout) << "isedge,1 = " << isedge << "\n";
+		    // (*testout) << "isedge,1 = " << isedge << "\n";
 
-		  // there must exist at least two different faces on edge
-		  if (isedge)
-		    {
-		      int cnts = 0;
-		      for (m = 0; m < surfind.Size(); m++)
-			{
-			  if (fabs (normalvecs[m] * t) > 1e-6)
-			    continue;
-
-			  Vec<3> s = Cross (normalvecs[m], t);
-			  Vec<3> t2a = t + 0.01 *s;
-			  Vec<3> t2b = t - 0.01 *s;
-
-			  /*
-			  (*testout) << "nv = " << normalvecs[m] << ", s = " << s << "\n";
-			  (*testout) << "t2a = " << t2a << ", t2b = " << t2b << "\n";
-			  (*testout) << "via = "
-				     << locsol->VectorIn (p, t2a) << "/"
-				     << locsol->VectorStrictIn (p, t2a);
-			  (*testout) << "vib = "
-				     << locsol->VectorIn (p, t2b) << "/"
-				     << locsol->VectorStrictIn (p, t2b) << "\n";
-			  */
-
-			  bool isface =
-			    (locsol->VectorIn (p, t2a) &&
-			     !locsol->VectorStrictIn (p, t2a))
-			    ||
-			    (locsol->VectorIn (p, t2b) &&
-			     !locsol->VectorStrictIn (p, t2b));
+		    // there must exist at least two different faces on edge
+		    if (isedge)
+		      {
+			int cnts = 0;
+			for (int m = 0; m < surfind.Size(); m++)
+			  {
+			    if (fabs (normalvecs[m] * t) > 1e-6)
+			      continue;
+
+			    Vec<3> s = Cross (normalvecs[m], t);
+			    Vec<3> t2a = t + 0.01 *s;
+			    Vec<3> t2b = t - 0.01 *s;
+
+			    /*
+			      (*testout) << "nv = " << normalvecs[m] << ", s = " << s << "\n";
+			      (*testout) << "t2a = " << t2a << ", t2b = " << t2b << "\n";
+			      (*testout) << "via = "
+			      << locsol->VectorIn (p, t2a) << "/"
+			      << locsol->VectorStrictIn (p, t2a);
+			      (*testout) << "vib = "
+			      << locsol->VectorIn (p, t2b) << "/"
+			      << locsol->VectorStrictIn (p, t2b) << "\n";
+			    */
+
+			    bool isface =
+			      (locsol->VectorIn (p, t2a) &&
+			       !locsol->VectorStrictIn (p, t2a))
+			      ||
+			      (locsol->VectorIn (p, t2b) &&
+			       !locsol->VectorStrictIn (p, t2b));
 			  
-			  if (isface)
+			    if (isface)
+			      {
+				cnts++;
+			      }
+			  }
+			if (cnts < 2) isedge = 0;
+		      }
+
+		    if (isedge)
+		      {
+			int spi = -1;
+
+			searchtree.GetIntersecting (apoints[i]-Vec3d(1e-4,1e-4,1e-4), 
+						    apoints[i]+Vec3d(1e-4,1e-4,1e-4), 
+						    locsearch);
+
+			for (int m = 0; m < locsearch.Size(); m++)
+			  if (Dist2 (specpoints[locsearch[m]].p, apoints[i]) < 1e-8
+			      && Abs2(specpoints[locsearch[m]].v - t) < 1e-8)
 			    {
-			      cnts++;
+			      spi = locsearch[m];
+			      break;
 			    }
-			}
-		      if (cnts < 2) isedge = 0;
-		    }
 
-		  if (isedge)
-		    {
-		      spi = -1;
-		      for (m = 0; m < specpoints.Size(); m++)
-			if (Dist2 (specpoints[m].p, apoints[i]) < 1e-8
-			    && Abs2(specpoints[m].v - t) < 1e-8)
+
+			if (spi == -1)
 			  {
-			    spi = m;
-			    break;
+			    spi = specpoints.Append (SpecialPoint()) - 1;
+			    specpoint2point.Append (i);
+			    specpoints.Last().unconditional = 0;
+			    searchtree.Insert (apoints[i], spi);
 			  }
-		      if (spi == -1)
-			{
-			  spi = specpoints.Append (SpecialPoint()) - 1;
-			  specpoint2point.Append (i);
-			  specpoints.Last().unconditional = 0;
-			}
-		      specpoints[spi].p = apoints[i];
-		      specpoints[spi].v = t;
-		      if (surfind.Size() >= 3)
-			specpoints[spi].unconditional = 1;
-		      specpoints[spi].s1 = surfind[j];
-		      specpoints[spi].s2 = surfind[k];
-		      specpoints[spi].layer = apoints[i].GetLayer();
-		      for (int up = 0; up < geometry->GetNUserPoints(); up++)
-			if (Dist (geometry->GetUserPoint(up), apoints[i]) < 1e-10)
+
+			specpoints[spi].p = apoints[i];
+			specpoints[spi].v = t;
+			if (surfind.Size() >= 3)
 			  specpoints[spi].unconditional = 1;
-			
-		      /*
-		      (*testout) << "spi = " << spi 
-				 << " uncond = " << specpoints[spi].unconditional
-				 << " t = " << t << "\n";
-		      */
-		    }
+			specpoints[spi].s1 = surfind[j];
+			specpoints[spi].s2 = surfind[k];
+			specpoints[spi].layer = apoints[i].GetLayer();
+			for (int up = 0; up < geometry->GetNUserPoints(); up++)
+			  if (Dist (geometry->GetUserPoint(up), apoints[i]) < 1e-10)
+			    specpoints[spi].unconditional = 1;
+		      }
           
-		}
-	  delete locsol;
-	}
-    }
+		  }
+	    delete locsol;
+	  }
+      }
 
-  // if special point is unconditional on some solid,
-  // it must be unconditional everywhere:
+    // if special point is unconditional on some solid,
+    // it must be unconditional everywhere:
 
-  BitArray uncond (apoints.Size());
-  uncond.Clear();
+    BitArray uncond (apoints.Size());
+    uncond.Clear();
 
-  for (i = 0; i < specpoints.Size(); i++)
-    if (specpoints[i].unconditional)
-      uncond.Set (specpoint2point[i]);
+    for (int i = 0; i < specpoints.Size(); i++)
+      if (specpoints[i].unconditional)
+	uncond.Set (specpoint2point[i]);
   
-  for (i = 0; i < specpoints.Size(); i++)
-    specpoints[i].unconditional = 
-      uncond.Test (specpoint2point[i]) ? 1 : 0;
-}
+    for (int i = 0; i < specpoints.Size(); i++)
+      specpoints[i].unconditional = 
+	uncond.Test (specpoint2point[i]) ? 1 : 0;
+  }
 }
diff --git a/Netgen/libsrc/csg/specpoin.hpp b/Netgen/libsrc/csg/specpoin.hpp
index 244361e652..8fd26f3207 100644
--- a/Netgen/libsrc/csg/specpoin.hpp
+++ b/Netgen/libsrc/csg/specpoin.hpp
@@ -3,7 +3,7 @@
 
 
 /**************************************************************************/
-/* File:   specpoin.hh                                                    */
+/* File:   specpoin.hpp                                                   */
 /* Author: Joachim Schoeberl                                              */
 /* Date:   01. Okt. 95                                                    */
 /**************************************************************************/
@@ -21,7 +21,6 @@ class Solid;
 class SpecialPoint
 {
 public:
-  
   /// coordinates
   Point<3> p;
   /// tangential to edge
@@ -35,7 +34,7 @@ public:
   int s1, s2;        
 
   ///
-  SpecialPoint () : p(), v(), layer(0)
+  SpecialPoint () : p(0,0,0), v(0,0,0), layer(0), unconditional(0), s1(0), s2(0)
   { ; }
 
   ///
@@ -51,7 +50,10 @@ public:
   int GetLayer() const { return layer; }
 
   ///
-  bool HasSurfaces (int as1, int as2) const;
+  bool HasSurfaces (int as1, int as2) const
+  {
+    return (s1 == as1 && s2 == as2 || s1 == as2 && s2 == as1);
+  }
 };
 
 
@@ -119,7 +121,7 @@ protected:
 
 
 
-
+  /*
   ///
   bool ExtremalPointPossible (const Surface * f1, const Surface * f2, 
 			      int dir, const BoxSphere<3> & box);
@@ -129,6 +131,7 @@ protected:
   ///
   bool ExtremalPointNewtonConvergence (const Surface * f1, const Surface * f2, 
 				       int dir, const BoxSphere<3> & box);
+  */
   ///
   void ExtremalPointNewton (const Surface * f1, const Surface * f2, 
 			    int dir, Point<3> & p);
@@ -136,6 +139,10 @@ protected:
 
   ///
   bool AddPoint (const Point<3> & p, int layer);
+
+  void ComputeExtremalPoints (const Plane * plane, 
+			      const QuadraticSurface * quadric, 
+			      ARRAY<Point<3> > & pts);
 };
 
 #endif
diff --git a/Netgen/libsrc/csg/specpoin_new.cpp b/Netgen/libsrc/csg/specpoin_new.cpp
new file mode 100644
index 0000000000..4ac2c130fa
--- /dev/null
+++ b/Netgen/libsrc/csg/specpoin_new.cpp
@@ -0,0 +1,1367 @@
+#include <mystdlib.h>
+#include <meshing.hpp>
+#include <csg.hpp>
+
+
+/*
+   Special Point calculation uses the global Flags:
+
+   size .. 500       cube = [-size, size]^3
+   relydegtest       when to rely on degeneration ?
+   calccp            calculate points of intersection ?
+   cpeps1            eps for degenerated poi
+   calcep            calculate points of extreme coordinates ?
+   epeps1            eps for degenerated edge
+   epeps2            eps for axis parallel pec
+   epspointdist      eps for distance of special points 
+*/
+
+
+namespace netgen
+{
+void ProjectToEdge (const Surface * f1, const Surface * f2, Point<3> & hp);
+
+
+
+SpecialPoint :: SpecialPoint (const SpecialPoint & sp)
+{
+  p = sp.p;
+  v = sp.v;
+  s1 = sp.s1;
+  s2 = sp.s2;
+  layer = sp.layer;
+  unconditional = sp.unconditional;
+}
+  
+SpecialPoint & SpecialPoint :: operator= (const SpecialPoint & sp)
+{
+  p = sp.p;
+  v = sp.v;
+  s1 = sp.s1;
+  s2 = sp.s2;
+  layer = sp.layer;
+  unconditional = sp.unconditional;
+  return *this;
+}
+
+
+void SpecialPoint :: Print (ostream & str)
+{
+  str << "p = " << p << "   v = " << v 
+      << " s1/s2 = " << s1 << "/" << s2
+      << " layer = " << layer
+      << endl;
+}
+
+
+
+
+SpecialPointCalculation :: SpecialPointCalculation ()
+{
+  ;  
+}
+
+void SpecialPointCalculation :: 
+CalcSpecialPoints (const CSGeometry & ageometry, 
+		   ARRAY<MeshPoint> & apoints)
+{
+  int i;
+
+  geometry = &ageometry;
+  points = &apoints;
+
+  size = geometry->MaxSize(); 
+  (*testout) << "Find Special Points" << endl;
+  (*testout) << "maxsize = " << size << endl;
+
+  cpeps1 = 1e-6; 
+  epeps1 = 1e-3; 
+  epeps2 = 1e-6; 
+
+  epspointdist2 = sqr (size * 1e-8); 
+  relydegtest = size * 1e-4; 
+
+
+  BoxSphere<3> box (Point<3> (-size, -size, -size),
+		    Point<3> ( size,  size,  size));
+
+  box.CalcDiamCenter();
+  PrintMessage (3, "main-solids: ", geometry->GetNTopLevelObjects());
+
+  for (i = 0; i < geometry->GetNTopLevelObjects(); i++)
+    {
+      (*testout) << "tlo " << i << ":" << endl;
+      const TopLevelObject * tlo = geometry->GetTopLevelObject(i);
+      tlo->GetSolid()->Print (*testout); 
+      (*testout) << endl;
+      CalcSpecialPointsRec (tlo->GetSolid(), tlo->GetLayer(),
+			    box, 1, 1, 1);
+    }
+  
+  PrintDot ('\n');
+
+
+  // add user point:
+  for (i = 0; i < geometry->GetNUserPoints(); i++)
+    AddPoint (geometry->GetUserPoint(i), 1);
+  
+  PrintMessage (3, apoints.Size(), " special points");
+
+  for (i = 0; i < boxesinlevel.Size(); i++)
+    (*testout) << "level " << i << " has " 
+	       << boxesinlevel[i] << " boxes" << endl;
+}
+
+
+
+// int debug;
+void SpecialPointCalculation :: 
+CalcSpecialPointsRec (const Solid * sol, int layer,
+		      const BoxSphere<3> & box, 
+		      int level, bool calccp, bool calcep)
+{
+  if (multithread.terminate)
+    return;
+
+  int i;
+  BoxSphere<3> sbox;
+  Solid * redsol;
+
+  int numprim;
+
+  bool decision;
+  bool possiblecrossp, possibleexp;  // possible cross or extremalpoint
+  bool surecrossp, sureexp;          // sure ...
+  
+  static ARRAY<int> locsurf;  // attention: array is static
+
+
+  Point<3> p;
+  int k1, k2, k3;
+  int extremdir;
+  double hd;
+
+  if (!sol) return;
+
+  if (level >= 100)
+    {
+      cerr << "Problems in CalcSpecialPoints" << endl;
+      cerr << "Point: " << box.Center() << endl;
+      exit (1);
+    }
+
+  static int cntbox = 0;
+  cntbox++;
+  if (cntbox % 10000 == 0) 
+    PrintDot ();
+
+  if (level <= boxesinlevel.Size())
+    boxesinlevel.Elem(level)++;
+  else
+    boxesinlevel.Append (1);
+
+  /*
+  numprim = sol -> NumPrimitives();
+  sol -> GetSurfaceIndices (locsurf);
+  */
+
+  // debug = 0;
+  // box.IsIn (Point<3> (4.9, 1.279, 2.8));
+
+
+  geometry -> GetIndependentSurfaceIndices (sol, box, locsurf);
+  numprim = locsurf.Size();
+
+  /*  
+  if (debug)
+    {
+      (*testout) << "box = " << box.PMin() << "-" << box.PMax()
+		 << " np = " << numprim << " : ";
+      for (i = 1; i <= locsurf.Size(); i++)
+	(*testout) << " " << locsurf.Get(i);
+      (*testout) << " diam = " << box.Diam();
+      (*testout) << " numprim = " << numprim;
+      (*testout) << endl;
+    }
+  */
+
+
+  p = box.Center();
+  
+  /*
+  (*testout) << "box = " << box.PMin() << " - " << box.PMax()
+	     << ", lev = " << level 
+	     << ", nprim = " << sol->NumPrimitives() 
+	     << ", lsurf = " << locsurf.Size() << endl;
+
+  for (i = 1; i <= locsurf.Size(); i++)
+    geometry->GetSurface (locsurf.Get(i)) -> Print (*testout);
+  sol -> Print (*testout);
+  */
+
+  /*
+    for (i = 1; i <= locsurf.Size(); i++)
+    (*testout) << locsurf.Get(i) << " ";
+    (*testout) << "C = " << box.Center() << " diam = " << box.Diam() << endl;
+    */
+
+  possiblecrossp = (numprim >= 3) && calccp;
+  surecrossp = 0;
+
+  if (possiblecrossp && (locsurf.Size() <= 10))
+    {
+      decision = 1;
+      surecrossp = 0;
+
+      for (k1 = 1; k1 <= locsurf.Size() - 2; k1++)
+	for (k2 = k1 + 1; k2 <= locsurf.Size() - 1; k2++)
+	  for (k3 = k2 + 1; k3 <= locsurf.Size(); k3++)
+	    {
+	      int nc, deg;
+	      nc = CrossPointNewtonConvergence 
+		(geometry->GetSurface(locsurf.Get(k1)), 
+		 geometry->GetSurface(locsurf.Get(k2)), 
+		 geometry->GetSurface(locsurf.Get(k3)), p );
+	      
+	      deg = CrossPointDegenerated 
+		(geometry->GetSurface(locsurf.Get(k1)), 
+		 geometry->GetSurface(locsurf.Get(k2)), 
+		 geometry->GetSurface(locsurf.Get(k3)), box );
+	      
+	      if (!nc && !deg) decision = 0;
+	      if (nc) surecrossp = 1;
+	    }
+
+      if (decision && surecrossp)
+	{
+	  for (k1 = 1; k1 <= locsurf.Size() - 2; k1++)
+	    for (k2 = k1 + 1; k2 <= locsurf.Size() - 1; k2++)
+	      for (k3 = k2 + 1; k3 <= locsurf.Size(); k3++)
+		{
+		  if (CrossPointNewtonConvergence 
+		      (geometry->GetSurface(locsurf.Get(k1)), 
+		       geometry->GetSurface(locsurf.Get(k2)), 
+		       geometry->GetSurface(locsurf.Get(k3)), p ) )
+		    {
+		      Point<3> pp = p;
+		      CrossPointNewton 
+			(geometry->GetSurface(locsurf.Get(k1)), 
+			 geometry->GetSurface(locsurf.Get(k2)), 
+			 geometry->GetSurface(locsurf.Get(k3)), pp);
+              
+		      BoxSphere<3> hbox (pp, pp);
+		      hbox.Increase (1e-8);
+
+		      if (pp(0) > box.PMin()(0) - 1e-5 && 
+			  pp(0) < box.PMax()(0) + 1e-5 &&
+			  pp(1) > box.PMin()(1) - 1e-5 && 
+			  pp(1) < box.PMax()(1) + 1e-5 &&
+			  pp(2) > box.PMin()(2) - 1e-5 && 
+			  pp(2) < box.PMax()(2) + 1e-5 &&
+			  sol -> IsIn (pp) && !sol->IsStrictIn (pp) &&
+			  !CrossPointDegenerated
+			  (geometry->GetSurface(locsurf.Get(k1)), 
+			   geometry->GetSurface(locsurf.Get(k2)), 
+			   geometry->GetSurface(locsurf.Get(k3)), hbox ))
+
+			{ 
+			  //                AddCrossPoint (locsurf, sol, p);
+			  BoxSphere<3> boxp (pp, pp);
+			  boxp.Increase (1e-3);
+			  boxp.CalcDiamCenter();
+			  ARRAY<int> locsurf2;
+
+			  geometry -> GetIndependentSurfaceIndices (sol, boxp, locsurf2);
+			  
+			  /*
+			  ReducePrimitiveIterator rpi(boxp);
+			  UnReducePrimitiveIterator urpi;
+			  
+			  ((Solid*)sol) -> IterateSolid (rpi);
+			  sol -> GetIndependentSurfaceIndices (locsurf2);
+			  ((Solid*)sol) -> IterateSolid (urpi);
+			  */
+			  bool found1 = 0, found2 = 0, found3 = 0;
+			  for (i = 1; i <= locsurf2.Size(); i++)
+			    {
+			      if (locsurf2.Get(i) == locsurf.Get(k1))
+				found1 = 1;
+			      if (locsurf2.Get(i) == locsurf.Get(k2))
+				found2 = 1;
+			      if (locsurf2.Get(i) == locsurf.Get(k3))
+				found3 = 1;
+			    }
+			  
+			  if (found1 && found2 && found3)
+			    if (AddPoint (pp, layer))
+			      {
+				(*testout) << "Crosspoint found: " << pp 
+					   << " diam = " << box.Diam() << endl;
+				(*testout) << "surfs: " 
+					   << locsurf.Get(k1) << "," 
+					   << locsurf.Get(k2) << "," 
+					   << locsurf.Get(k3) << endl;
+			      }
+			}
+		    }
+		}
+	}
+      
+      if (decision)
+	possiblecrossp = 0;
+    }
+
+
+
+
+  possibleexp = (numprim >= 2) && calcep;
+    
+  if (possibleexp && (locsurf.Size() <= 10))
+    {
+      decision = 1;
+      sureexp = 0;
+
+      for (k1 = 1; k1 <= locsurf.Size() - 1; k1++)
+	for (k2 = k1+1; k2 <= locsurf.Size(); k2++)
+	  {
+	    bool nc, deg;
+	    
+	    nc = EdgeNewtonConvergence 
+	      (geometry->GetSurface(locsurf.Get(k1)),
+	       geometry->GetSurface(locsurf.Get(k2)),
+	       p);
+	    
+	    deg = EdgeDegenerated 
+	      (geometry->GetSurface(locsurf.Get(k1)),
+	       geometry->GetSurface(locsurf.Get(k2)),
+	       box);
+	    
+	    if (!nc && !deg) decision = 0;
+	    if (nc) sureexp = 1;
+
+	    /*
+	    if (debug)
+	      {
+		(*testout) << "p = " << p << " s1,2 = " << locsurf.Get(k1) << ", " << locsurf.Get(k2) 
+			   << " nc = " << nc << " deg = " << deg << endl;
+	      }
+	    */
+	  }
+
+      if (decision && sureexp)
+	{
+	  for (k1 = 1; k1 <= locsurf.Size() - 1; k1++)
+	    for (k2 = k1+1; k2 <= locsurf.Size(); k2++)
+	      {
+		if (
+		    EdgeNewtonConvergence 
+		    (geometry->GetSurface(locsurf.Get(k1)),
+		     geometry->GetSurface(locsurf.Get(k2)),
+		     p) )
+		  {
+		    EdgeNewton 
+		      (geometry->GetSurface(locsurf.Get(k1)),
+		       geometry->GetSurface(locsurf.Get(k2)),
+		       p);
+		    
+		    Point<3> pp;
+		    if (IsEdgeExtremalPoint 
+		      (geometry->GetSurface(locsurf.Get(k1)),
+		       geometry->GetSurface(locsurf.Get(k2)),
+		       p, pp, box.Diam()/2))
+			{
+			  (*testout) << "extremalpoint (nearly) found:" 
+				     << pp
+				     << endl;
+			  if (Dist (pp, box.Center()) < box.Diam()/2 &&
+			      sol -> IsIn (pp) && !sol->IsStrictIn (pp) )
+			    {
+			      //                AddExtremalPoint (locsurf.Get(k1), locsurf.Get(k2), p);
+			      if (AddPoint (pp, layer))
+				(*testout) << "Extremal point found: " << pp << endl;
+			    }  
+			}            
+		  }
+	      }
+	}
+      if (decision)
+	possibleexp = 0;
+    }
+ 
+
+
+  if (possiblecrossp || possibleexp)
+    {
+      for (i = 0; i < 8; i++)
+	{
+	  box.GetSubBox (i, sbox);
+	  sbox.Increase (1e-4 * sbox.Diam());
+
+	  redsol = sol -> GetReducedSolid (sbox);
+
+	  if (redsol)
+	    {
+	      CalcSpecialPointsRec (redsol, layer, sbox, level+1, calccp, calcep);
+	      delete redsol;
+	    }
+	}
+    }
+}
+
+
+
+
+
+/******* Tests for Point of intersection **********************/
+
+
+
+bool SpecialPointCalculation :: 
+CrossPointNewtonConvergence (const Surface * f1, 
+			     const Surface * f2, 
+			     const Surface * f3,
+			     const Point<3> & p)
+{
+  int i;
+  Vec<3> grad;
+  Vec<3> rs, x;
+  Mat<3> jacobi, inv;
+  double alpha, beta, gamma, eta;
+  double sum;
+  int j;
+
+
+  rs(0) = f1->CalcFunctionValue (p);
+  rs(1) = f2->CalcFunctionValue (p);
+  rs(2) = f3->CalcFunctionValue (p);
+
+  f1->CalcGradient (p, grad);
+  jacobi(0,0) = grad(0);
+  jacobi(0,1) = grad(1);
+  jacobi(0,2) = grad(2);
+
+  f2->CalcGradient (p, grad);
+  jacobi(1,0) = grad(0);
+  jacobi(1,1) = grad(1);
+  jacobi(1,2) = grad(2);
+
+  f3->CalcGradient (p, grad);
+  jacobi(2,0) = grad(0);
+  jacobi(2,1) = grad(1);
+  jacobi(2,2) = grad(2);
+
+  alpha = 1;
+  if (fabs (Det (jacobi)) > 1e-8)
+    {
+      CalcInverse (jacobi, inv);
+      x = inv * rs;
+
+      gamma = f1 -> HesseNorm() + f2 -> HesseNorm() + f3 -> HesseNorm();
+      beta = 0;
+      for (i = 0; i < 3; i++)
+	{
+	  sum = 0;
+	  for (j = 0; j < 3; j++)
+	    sum += fabs (inv(i,j));
+	  beta = max2 (beta, sum);
+	}
+      eta = Abs (x);
+
+      alpha = beta * gamma * eta;
+    }
+
+  return (alpha < 0.1);
+}
+
+
+
+
+bool SpecialPointCalculation :: 
+CrossPointDegenerated (const Surface * f1,
+		       const Surface * f2, 
+		       const Surface * f3, 
+		       const BoxSphere<3> & box) const
+{
+  Mat<3> mat;
+  Vec<3> grad, g1, g2, g3;
+  double normprod;
+
+  if (box.Diam() > relydegtest) return 0;
+
+  f1->CalcGradient (box.Center(), g1);
+  normprod = Abs (g1);
+
+  f2->CalcGradient (box.Center(), g2);
+  normprod *= Abs (g2);
+ 
+  f3->CalcGradient (box.Center(), g3);
+  normprod *= Abs (g3);
+
+  for (int i = 0; i < 3; i++)
+    {
+      mat(i,0) = g1(i);
+      mat(i,1) = g2(i);
+      mat(i,2) = g3(i);
+    }
+
+  if (fabs (Det (mat)) < cpeps1 * normprod)
+    return 1;
+  else 
+    return 0;
+}
+ 
+
+
+
+
+void SpecialPointCalculation :: CrossPointNewton (const Surface * f1, 
+						  const Surface * f2, 
+						  const Surface * f3, Point<3> & p)
+{
+  int i;
+  Vec<3> g1, g2, g3;
+  Vec<3> rs, sol;
+  Mat<3> mat;
+
+  i = 10;
+  while (i > 0)
+    {
+      i--;
+      rs(0) = f1->CalcFunctionValue (p);
+      rs(1) = f2->CalcFunctionValue (p);
+      rs(2) = f3->CalcFunctionValue (p);
+
+      f1->CalcGradient (p, g1);
+      f2->CalcGradient (p, g2);
+      f3->CalcGradient (p, g3);
+
+      for (int j = 0; j < 3; j++)
+	{
+	  mat(0, j) = g1(j);
+	  mat(1, j) = g2(j);
+	  mat(2, j) = g3(j);
+	}
+      mat.Solve (rs, sol);
+	  /*
+      Transpose (g1, g2, g3);
+      SolveLinearSystem (g1, g2, g3, rs, sol);
+	  */
+      if (sol.Length() < 1e-12 && i > 1) i = 1;
+
+      p -= sol;
+    }
+}
+
+
+
+
+/******* Tests for Point on edges **********************/
+
+
+
+
+bool SpecialPointCalculation :: 
+EdgeNewtonConvergence (const Surface * f1, const Surface * f2, 
+		       const Point<3> & p)
+{
+  int i;
+  Vec<3> g1, g2, sol;
+  Vec<2> vrs;
+  double alpha, beta, gamma, eta;
+  double sum;
+  Mat<2,3> mat;
+  Mat<3,2> inv;
+  int j;
+
+  vrs(0) = f1->CalcFunctionValue (p);
+  vrs(1) = f2->CalcFunctionValue (p);
+
+  f1->CalcGradient (p, g1);
+  f2->CalcGradient (p, g2);
+
+  for (i = 0; i < 3; i++)
+    {
+      mat(0,i) = g1(i);
+      mat(1,i) = g2(i);
+    }
+
+  alpha = 1;
+
+  if ( fabs(g1 * g2) < (1 - 1e-8) * Abs (g1) * Abs (g2))
+    {
+      CalcInverse (mat, inv);
+      sol = inv * vrs;
+
+      // SolveLinearSystemLS (g1, g2, vrs, sol);
+
+      gamma = f1 -> HesseNorm() + f2 -> HesseNorm();
+
+      /*
+      Vec<3> inv1, inv2;
+      PseudoInverse (g1, g2, inv1, inv2);
+      */
+      
+      beta = 0;
+      for (i = 0; i < 3; i++)
+	for (j = 0; j < 2; j++)
+	  beta += inv(i,j) * inv(i,j);
+      beta = sqrt (beta);
+
+      //      beta = inv1.Length() + inv2.Length();
+      eta = Abs (sol);
+      alpha = beta * gamma * eta;
+    }
+  return (alpha < 0.1);
+}
+
+
+
+
+bool SpecialPointCalculation :: 
+EdgeDegenerated (const Surface * f1,
+		 const Surface * f2, 
+		 const BoxSphere<3> & box) const
+{
+  // perform newton steps. normals parallel ?
+  // if not decideable: return 0 
+  
+  
+  Point<3> p = box.Center();
+  int i;
+  Vec<3> grad, g1, g2, sol;
+  Vec<2> vrs;
+  Mat<2,3> mat;
+
+  i = 20;
+  while (i > 0)
+    {
+      if (Dist (p, box.Center()) > box.Diam())
+	return 0;
+
+      i--;
+      vrs(0) = f1->CalcFunctionValue (p);
+      vrs(1) = f2->CalcFunctionValue (p);
+
+      f1->CalcGradient (p, g1);
+      f2->CalcGradient (p, g2);
+
+      if ( fabs (g1 * g2) > (1 - 1e-10) * Abs (g1) * Abs (g2))
+	return 1;
+
+      for (int j = 0; j < 3; j++)
+	{
+	  mat(0,j) = g1(j);
+	  mat(1,j) = g2(j);
+	}
+      mat.Solve (vrs, sol);
+      //      SolveLinearSystemLS (g1, g2, vrs, sol);
+
+      if (Abs (sol) < 1e-12 && i > 1) i = 1;
+      p -= sol;
+    }
+
+  return 0;
+}
+
+
+
+
+
+
+void SpecialPointCalculation :: EdgeNewton (const Surface * f1, 
+					    const Surface * f2, Point<3> & p)
+{
+  int i;
+  Vec<3> grad, g1, g2, sol;
+  Vec<2> vrs;
+  Mat<2,3> mat;
+
+  i = 10;
+  while (i > 0)
+    {
+      i--;
+      vrs(0) = f1->CalcFunctionValue (p);
+      vrs(1) = f2->CalcFunctionValue (p);
+
+      f1->CalcGradient (p, g1);
+      f2->CalcGradient (p, g2);
+
+      for (int j = 0; j < 3; j++)
+	{
+	  mat(0,j) = g1(j);
+	  mat(1,j) = g2(j);
+	}
+      mat.Solve (vrs, sol);
+      //      SolveLinearSystemLS (g1, g2, vrs, sol);
+
+      if (Abs (sol) < 1e-12 && i > 1) i = 1;
+      p -= sol;
+    }
+}
+
+
+
+bool SpecialPointCalculation :: 
+IsEdgeExtremalPoint (const Surface * f1, const Surface * f2, 
+		     const Point<3> & p, Point<3> & pp, double rad)
+{
+  Vec<3> g1, g2, t, t1, t2;
+  int j;
+
+  f1->CalcGradient (p, g1);
+  f2->CalcGradient (p, g2);
+  
+  t = Cross (g1, g2);
+  t.Normalize();
+
+  Point<3> p1 = p + rad * t;
+  Point<3> p2 = p - rad * t;
+
+  EdgeNewton (f1, f2, p1);
+  EdgeNewton (f1, f2, p2);
+
+  
+  f1->CalcGradient (p1, g1);
+  f2->CalcGradient (p1, g2);
+  t1 = Cross (g1, g2);
+  t1.Normalize();
+
+  f1->CalcGradient (p2, g1);
+  f2->CalcGradient (p2, g2);
+  t2 = Cross (g1, g2);
+  t2.Normalize();
+
+  double val = 1e-8 * rad * rad;
+  for (j = 0; j < 3; j++)
+    if ( (t1(j) * t2(j) < -val) )
+      {
+	pp = p;
+	ExtremalPointNewton (f1, f2, j+1, pp);
+	return 1;
+      }
+
+  return 0;
+}
+
+
+
+
+
+
+
+
+
+/********** Tests of Points of extremal coordinates  ****************/
+
+
+void SpecialPointCalculation :: ExtremalPointNewton (const Surface * f1, 
+						     const Surface * f2, 
+						     int dir, Point<3> & p)
+{
+  int i;
+
+  Vec<3> g1, g2, v, curv;
+  Vec<3> rs, x, y1, y2, y;
+  Mat<3> h1, h2;
+  Mat<3> jacobi;
+
+
+  if (dir < 1 || dir > 3)
+    {
+      cerr << "Error: Illegal extremdir" << endl;
+      return;
+    }
+
+  i = 50;
+  while (i > 0)
+    {
+      i--;
+      rs(0) = f1->CalcFunctionValue (p);
+      rs(1) = f2->CalcFunctionValue (p);
+
+      f1 -> CalcGradient (p, g1);
+      f2 -> CalcGradient (p, g2);
+
+      f1 -> CalcHesse (p, h1);
+      f2 -> CalcHesse (p, h2);
+
+
+      v = Cross (g1, g2);
+
+      rs(2) = v(dir-1);
+
+      jacobi(0,0) = g1(0);
+      jacobi(0,1) = g1(1);
+      jacobi(0,2) = g1(2);
+
+      jacobi(1,0) = g2(0);
+      jacobi(1,1) = g2(1);
+      jacobi(1,2) = g2(2);
+
+
+      switch (dir)
+	{
+	case 1:
+	  {
+	    y1(0) = 0;
+	    y1(1) = g2(2);
+	    y1(2) = -g2(1);
+	    y2(0) = 0;
+	    y2(1) = -g1(2);
+	    y2(2) = g1(1);
+	    break;
+	  }
+	case 2:
+	  {
+	    y1(0) = -g2(2);
+	    y1(1) = 0;
+	    y1(2) = g2(0);
+	    y2(0) = g1(2);
+	    y2(1) = 0;
+	    y2(2) = -g1(0);
+	    break;
+	  }
+	case 3:
+	  {
+	    y1(0) = g2(1);
+	    y1(1) = -g2(0);
+	    y1(2) = 0;
+	    y2(0) = -g1(1);
+	    y2(1) = g1(0);
+	    y2(2) = 0;
+	    break;
+	  }
+	}
+
+      y = h1 * y1 + h2 * y2;
+
+      jacobi(2,0) = y(0);
+      jacobi(2,1) = y(1);
+      jacobi(2,2) = y(2);
+
+      jacobi.Solve (rs, x);
+      /*
+      CalcInverse (jacobi, inv);
+      inv.Mult (rs, x);
+      */
+      //    (*testout) << "err = " << x.L2Norm() << endl;
+
+      if (Abs (x) < 1e-12 && i > 1)
+	{
+	  //      (*testout) << "convergent in " << (10 - i) << " steps " << endl;
+
+	  i = 1;
+	}
+      
+      p -= x;
+    }
+
+  if (Abs (x) > 1e-10)
+    {
+      (*testout) << "Error: extremum Newton not convergent" << endl;
+      (*testout) << "dir = " << dir << endl;
+      (*testout) << "p = " << p << endl;
+      (*testout) << "x = " << x << endl;
+    }
+}
+
+
+
+
+bool SpecialPointCalculation :: ExtremalPointPossible (const Surface * f1, 
+						       const Surface * f2, 
+						       int dir, 
+						       const BoxSphere<3> & box)
+{
+  double hn1, hn2, gn1, gn2;
+  Point<3> p;
+  Vec<3> g1, g2, v;
+  double f3;
+  double r = box.Diam()/2;
+
+  p = box.Center();
+
+  f1 -> CalcGradient (p, g1);
+  f2 -> CalcGradient (p, g2);
+
+  gn1 = g1.Length();
+  gn2 = g2.Length();
+
+  hn1 = f1 -> HesseNorm ();
+  hn2 = f2 -> HesseNorm ();
+
+  v = Cross (g1, g2);
+  f3 = fabs (v(dir-1));
+
+  //  (*testout) << "f3 = " << f3 << "  r = " << r 
+  //             << "normbound = " 
+  //             << (hn1 * (gn2 + r * hn2) + hn2 * (gn1 + r * hn1)) << endl;
+ 
+  return (f3 <= 3 * r * (hn1 * (gn2 + r * hn2) + hn2 * (gn1 + r * hn1)));
+}
+
+
+
+bool SpecialPointCalculation :: 
+ExtremalPointNewtonConvergence (const Surface * f1, const Surface * f2, 
+				int dir, 
+				const BoxSphere<3> & box)
+{
+  return box.Diam() < 1e-8;
+}
+
+
+bool SpecialPointCalculation :: 
+ExtremalPointDegenerated (const Surface * f1, const Surface * f2, 
+			  int dir, const BoxSphere<3> & box)
+{
+  double gn1, gn2;
+  Point<3> p;
+  Vec<3> g1, g2, v;
+  double maxderiv;
+  double minv;
+  Vec<3> curv, t;
+  Vec<2> rs, x;
+  Mat<3> h1, h2;
+  Mat<2> a, inv;
+  double leftside;
+
+  if (box.Diam() > relydegtest) return 0;
+
+  p = box.Center();
+
+  f1 -> CalcGradient (p, g1);
+  f2 -> CalcGradient (p, g2);
+  gn1 = g1.Length();
+  gn2 = g2.Length();
+
+  v = Cross (g1, g2);
+  if (Abs (v) < epeps1 * gn1 * gn2) return 1;       // irregular edge
+
+  f1 -> CalcHesse (p, h1);
+  f2 -> CalcHesse (p, h2);
+
+  //  hn1 = f1 -> HesseNorm ();
+  //  hn2 = f2 -> HesseNorm ();
+
+  t = v;
+  a(0, 0) = g1 * g1;
+  a(0, 1) = 
+    a(1, 0) = g1 * g2;
+  a(1, 1) = g2 * g2;
+  
+  rs(0) = g1(dir-1);
+  rs(1) = g2(dir-1);
+
+  a.Solve (rs, x);
+
+  /*
+  CalcInverse (a, inv);
+  inv.Mult (rs, x);          // x .. Lagrangeparameter
+  */
+  //  (*testout) << "g1 = " << g1 << " g2 = " << g2 << endl;
+  //  (*testout) << "lam = " << x << endl;
+  //  (*testout) << "h2 = " << h2 << endl;
+
+  leftside = fabs (x(0) * ( t * (h1 * t)) + 
+                   x(1) * ( t * (h2 * t)));
+
+  //  (*testout) << "leftside = " << leftside << endl;
+
+  if (leftside < epeps2 * Abs2 (v)) return 1;  
+
+  return 0;
+}
+
+ 
+
+bool SpecialPointCalculation :: AddPoint (const Point<3> & p, int layer)
+{
+  for (int i = 0; i < points->Size(); i++)
+    if (Dist2 ( (*points)[i], p) < epspointdist2 &&
+	(*points)[i].GetLayer() == layer)
+      return 0;
+
+  points->Append (MeshPoint(p, layer));
+  return 1;
+}
+
+
+
+
+
+
+
+/*
+void SpecialPointCalculation :: 
+AnalyzeSpecialPoints (const CSGeometry & ageometry,
+		      ARRAY<Point<3> > & apoints, 
+		      ARRAY<SpecialPoint> & specpoints)
+{
+  int si, i, j, k, l, m, spi;
+  Solid * locsol;
+  ARRAY<int> surfind;
+  ARRAY<Vec<3>> normalvecs;
+  const Solid * sol;
+  Vec<3> t;
+  Point<3> p;
+
+  ARRAY<int> specpoint2point;
+  specpoints.SetSize (0);
+ 
+  (*testout) << "AnalyzeSpecialPoints" << endl;
+
+  for (si = 1; si <= ageometry.GetNTopLevelObjects(); si++)
+    {
+      (*testout) << "main solid " << si << endl;
+
+      sol = ageometry.GetTopLevelObject(si)->GetSolid();
+      for (i = 1; i <= apoints.Size(); i++)
+	{
+	  p = apoints.Get(i);
+
+	  sol -> TangentialSolid (p, locsol);
+	  if (!locsol) continue;
+	  
+	  (*testout) << "Point " << apoints.Get(i) << endl;
+
+	  locsol -> GetSurfaceIndices (surfind);
+	  for (j = surfind.Size(); j >= 1; j--)
+	    if (fabs (ageometry.GetSurface(surfind.Get(j))->
+		      CalcFunctionValue (p)) > 1e-6)
+	      surfind.DeleteElement (j);
+
+
+
+	  (*testout) << "Surfaces: ";
+	  for (j = 1; j <= surfind.Size(); j++)
+	    (*testout) << surfind.Get(j) << " ";
+	  (*testout) << endl;
+
+	  normalvecs.SetSize(surfind.Size());
+	  for (j = 1; j <= surfind.Size(); j++)
+	    ageometry.GetSurface(surfind.Get(j)) -> 
+	      GetNormalVector(apoints.Get(i), normalvecs.Elem(j));
+
+	  for (j = 1; j <= normalvecs.Size() - 1; j ++)
+	    for (k = j+1; k <= normalvecs.Size(); k++)
+	      for (l = 1; l <= 2; l++)
+		{
+		  t = Cross (normalvecs.Get(j), normalvecs.Get(k));
+		  if (t.Length2() < 1e-8)
+		    {
+		      cerr << "AnalyzePoint: Surfaces degenerated" << endl;
+		      break;
+		    }
+		  t /= t.Length();
+		  if (l == 2) t *= -1;
+
+		  if (locsol -> Edge (apoints.Get(i), t))
+		    {
+		      spi = 0;
+		      for (m = 1; m <= specpoints.Size(); m++)
+			if (Dist2 (specpoints.Get(m).p, apoints.Get(i)) < 1e-8
+			    && (specpoints.Get(m).v - t).Length2() < 1e-8)
+			  {
+			    spi = m;
+			    break;
+			  }
+		      if (!spi)
+			{
+			  spi = specpoints.Append (SpecialPoint());
+			  specpoint2point.Append (i);
+			  specpoints.Last().unconditional = 0;
+			}
+		      specpoints.Elem(spi).p = apoints.Get(i);
+		      specpoints.Elem(spi).v = t;
+		      if (surfind.Size() >= 3)
+			specpoints.Elem(spi).unconditional = 1;
+		      specpoints.Elem(spi).s1 = surfind.Get(j);
+		      specpoints.Elem(spi).s2 = surfind.Get(k);
+		      (*testout) << "spi = " << spi 
+				 << " uncond = " << specpoints.Get(spi).unconditional
+				 << " t = " << t << endl;
+		    }
+          
+		}
+	  delete locsol;
+	}
+    }
+
+  // if special point is unconditional on some solid,
+  // it must be unconditional everywhere:
+
+  BitArray uncond (apoints.Size());
+  uncond.Clear();
+
+  for (i = 1; i <= specpoints.Size(); i++)
+    if (specpoints.Get(i).unconditional)
+      uncond.Set (specpoint2point.Get(i));
+
+  for (i = 1; i <= specpoints.Size(); i++)
+    specpoints.Elem(i).unconditional = 
+      uncond.Test (specpoint2point.Get(i)) ? 1 : 0;
+}
+*/
+
+
+
+void SpecialPointCalculation :: 
+AnalyzeSpecialPoints (const CSGeometry & ageometry,
+		      ARRAY<MeshPoint> & apoints, 
+		      ARRAY<SpecialPoint> & specpoints)
+{
+  int si, i, j, k, l, m, spi;
+
+  Solid * locsol;
+  ARRAY<int> surfind;
+  ARRAY<int> surfind2;
+
+  ARRAY<Vec<3> > normalvecs;
+  const Solid * sol;
+  const Surface * surf;
+
+  Vec<3> t, nsurf;
+  Point<3> p;
+
+  ARRAY<int> specpoint2point;
+  specpoints.SetSize (0);
+
+  geometry = &ageometry;
+ 
+  (*testout) << "AnalyzeSpecialPoints\n";
+
+
+  Box<3> bbox;
+  if (apoints.Size())
+    bbox.Set (apoints[0]);
+  for (int i = 1; i < apoints.Size(); i++)
+    bbox.Add (apoints[i]);
+  bbox.Increase (0.1 * Dist (bbox.PMin(), bbox.PMax()));
+
+  Point3dTree searchtree (bbox.PMin(), bbox.PMax());
+  ARRAY<int> locsearch;
+
+  for (si = 0; si < ageometry.GetNTopLevelObjects(); si++)
+    {
+      (*testout) << "main solid " << si << "\n";
+
+      sol = ageometry.GetTopLevelObject(si)->GetSolid();
+      surf = ageometry.GetTopLevelObject(si)->GetSurface();
+
+      for (i = 0; i < apoints.Size(); i++)
+	{
+	  p = apoints[i];
+	  if (ageometry.GetTopLevelObject(si)->GetLayer() !=
+	      apoints[i].GetLayer())
+	    continue;
+
+	  (*testout) << "Point " << apoints[i] << "\n";
+
+	  sol -> TangentialSolid (p, locsol);
+	  if (!locsol) continue;
+	  
+	  // get all surface indices, 
+	  if (surf)
+	    {
+	      locsol -> GetSurfaceIndices (surfind);
+	      bool hassurf = 0;
+	      for (m = 0; m < surfind.Size(); m++)
+		if (ageometry.GetSurface(surfind[m]) == surf)
+		  hassurf = 1;
+
+	      if (!hassurf)
+		continue;
+
+	      surf->GetNormalVector (p, nsurf);
+	    }
+
+	  // get independent surfaces of tangential solid
+
+	  BoxSphere<3> box(p,p);
+	  box.Increase (1e-6);
+	  box.CalcDiamCenter();
+	  ageometry.GetIndependentSurfaceIndices (locsol, box, surfind);
+
+
+	  (*testout) << "surfind.size = " << surfind.Size() << endl;
+
+	  /*
+	  locsol -> GetSurfaceIndices (surfind);
+	  for (j = surfind.Size(); j >= 1; j--)
+	    if (fabs (ageometry.GetSurface(surfind.Get(j))->
+		      CalcFunctionValue (p)) > 1e-6)
+	      surfind.DeleteElement (j);
+	  */
+
+	  /*
+	  (*testout) << "Surfaces: ";
+	  for (j = 0; j < surfind.Size(); j++)
+	    (*testout) << surfind[j] << " ";
+	  (*testout) << "\n";
+	  */
+
+
+	  normalvecs.SetSize(surfind.Size());
+	  for (j = 0; j < surfind.Size(); j++)
+	    ageometry.GetSurface(surfind[j]) ->
+	      GetNormalVector(apoints[i], normalvecs[j]);
+
+	  for (j = 0; j < normalvecs.Size(); j++)
+	    for (k = j+1; k < normalvecs.Size(); k++)
+	      for (l = 1; l <= 2; l++)
+		{
+		  t = Cross (normalvecs[j], normalvecs[k]);
+		  if (Abs2 (t) < 1e-8)
+		    {
+		      cerr << "AnalyzePoint: Surfaces degenerated" << "\n";
+		      break;
+		    }
+		  t.Normalize();
+		  if (l == 2) t *= -1;
+
+		  // try tangential direction t
+
+		  // (*testout) << "check tangential " << t << "\n";
+
+		  if (surf && fabs (nsurf * t) > 1e-6)
+		    continue;
+
+		  if (!surf)
+		    {
+		      ageometry.GetIndependentSurfaceIndices 
+			(locsol, p, t, surfind2);
+		  
+		      bool found1 = 0, found2 = 0;
+		      for (int ii = 0; ii < surfind2.Size(); ii++)
+			{
+			  if (surfind2[ii] == surfind[j])
+			    found1 = 1;
+			  if (surfind2[ii] == surfind[k])
+			    found2 = 1;
+			}
+		      if (!found1 || !found2)
+			continue;
+		    }
+
+
+		  bool isedge;
+
+		  // isedge = locsol -> Edge (apoints.Get(i), t);
+		  
+		  // edge must be on tangential surface
+		  isedge = 
+		    locsol->VectorIn (p, t) &&
+		    !locsol->VectorStrictIn (p, t);
+		  
+		  // (*testout) << "isedge,1 = " << isedge << "\n";
+
+		  // there must exist at least two different faces on edge
+		  if (isedge)
+		    {
+		      int cnts = 0;
+		      for (m = 0; m < surfind.Size(); m++)
+			{
+			  if (fabs (normalvecs[m] * t) > 1e-6)
+			    continue;
+
+			  Vec<3> s = Cross (normalvecs[m], t);
+			  Vec<3> t2a = t + 0.01 *s;
+			  Vec<3> t2b = t - 0.01 *s;
+
+			  /*
+			  (*testout) << "nv = " << normalvecs[m] << ", s = " << s << "\n";
+			  (*testout) << "t2a = " << t2a << ", t2b = " << t2b << "\n";
+			  (*testout) << "via = "
+				     << locsol->VectorIn (p, t2a) << "/"
+				     << locsol->VectorStrictIn (p, t2a);
+			  (*testout) << "vib = "
+				     << locsol->VectorIn (p, t2b) << "/"
+				     << locsol->VectorStrictIn (p, t2b) << "\n";
+			  */
+
+			  bool isface =
+			    (locsol->VectorIn (p, t2a) &&
+			     !locsol->VectorStrictIn (p, t2a))
+			    ||
+			    (locsol->VectorIn (p, t2b) &&
+			     !locsol->VectorStrictIn (p, t2b));
+			  
+			  if (isface)
+			    {
+			      cnts++;
+			    }
+			}
+		      if (cnts < 2) isedge = 0;
+		    }
+
+		  if (isedge)
+		    {
+		      spi = -1;
+
+		      searchtree.GetIntersecting (apoints[i]-Vec3d(1e-4,1e-4,1e-4), 
+						  apoints[i]+Vec3d(1e-4,1e-4,1e-4), 
+						  locsearch);
+		      
+		      for (m = 0; m < locsearch.Size(); m++)
+			if (Dist2 (specpoints[locsearch[m]].p, apoints[i]) < 1e-8
+			    && Abs2(specpoints[locsearch[m]].v - t) < 1e-8)
+			  {
+			    spi = locsearch[m];
+			    break;
+			  }
+
+		      /*
+		      for (m = 0; m < specpoints.Size(); m++)
+			if (Dist2 (specpoints[m].p, apoints[i]) < 1e-8
+			    && Abs2(specpoints[m].v - t) < 1e-8)
+			  {
+			    spi = m;
+			    break;
+			  }
+		      */
+		      if (spi == -1)
+			{
+			  spi = specpoints.Append (SpecialPoint()) - 1;
+			  specpoint2point.Append (i);
+			  specpoints.Last().unconditional = 0;
+			  searchtree.Insert (apoints[i], spi);
+			}
+		      specpoints[spi].p = apoints[i];
+		      specpoints[spi].v = t;
+		      if (surfind.Size() >= 3)
+			specpoints[spi].unconditional = 1;
+		      specpoints[spi].s1 = surfind[j];
+		      specpoints[spi].s2 = surfind[k];
+		      specpoints[spi].layer = apoints[i].GetLayer();
+		      for (int up = 0; up < geometry->GetNUserPoints(); up++)
+			if (Dist (geometry->GetUserPoint(up), apoints[i]) < 1e-10)
+			  specpoints[spi].unconditional = 1;
+			
+		      /*
+		      (*testout) << "spi = " << spi 
+				 << " uncond = " << specpoints[spi].unconditional
+				 << " t = " << t << "\n";
+		      */
+		    }
+          
+		}
+	  delete locsol;
+	}
+    }
+
+  // if special point is unconditional on some solid,
+  // it must be unconditional everywhere:
+
+  BitArray uncond (apoints.Size());
+  uncond.Clear();
+
+  for (i = 0; i < specpoints.Size(); i++)
+    if (specpoints[i].unconditional)
+      uncond.Set (specpoint2point[i]);
+  
+  for (i = 0; i < specpoints.Size(); i++)
+    specpoints[i].unconditional = 
+      uncond.Test (specpoint2point[i]) ? 1 : 0;
+}
+}
diff --git a/Netgen/libsrc/csg/specpoin_old.cpp b/Netgen/libsrc/csg/specpoin_old.cpp
new file mode 100644
index 0000000000..0f30ef7298
--- /dev/null
+++ b/Netgen/libsrc/csg/specpoin_old.cpp
@@ -0,0 +1,1370 @@
+#include <mystdlib.h>
+#include <meshing.hpp>
+#include <csg.hpp>
+
+
+/*
+
+   Special Point calculation uses the global Flags:
+
+
+   size .. 500       cube = [-size, size]^3
+   relydegtest       when to rely on degeneration ?
+   calccp            calculate points of intersection ?
+   cpeps1            eps for degenerated poi
+   calcep            calculate points of extreme coordinates ?
+   epeps1            eps for degenerated edge
+   epeps2            eps for axis parallel pec
+   epspointdist      eps for distance of special points 
+*/
+
+
+namespace netgen
+{
+void ProjectToEdge (const Surface * f1, const Surface * f2, Point<3> & hp);
+
+
+
+  /*
+SpecialPoint :: SpecialPoint ()
+  : p(), v(), layer(0)
+{
+  ;
+}
+  */
+
+
+SpecialPoint :: SpecialPoint (const SpecialPoint & sp)
+{
+  p = sp.p;
+  v = sp.v;
+  s1 = sp.s1;
+  s2 = sp.s2;
+  layer = sp.layer;
+  unconditional = sp.unconditional;
+}
+  
+SpecialPoint & SpecialPoint :: operator= (const SpecialPoint & sp)
+{
+  p = sp.p;
+  v = sp.v;
+  s1 = sp.s1;
+  s2 = sp.s2;
+  layer = sp.layer;
+  unconditional = sp.unconditional;
+  return *this;
+}
+
+/*
+bool SpecialPoint :: HasSurfaces (int as1, int as2) const
+{
+  return (s1 == as1 && s2 == as2 || s1 == as2 && s2 == as1);
+}
+*/
+void SpecialPoint :: Print (ostream & str)
+{
+  str << "p = " << p << "   v = " << v 
+      << " s1/s2 = " << s1 << "/" << s2
+      << " layer = " << layer
+      << endl;
+}
+
+
+
+
+SpecialPointCalculation :: SpecialPointCalculation ()
+{
+  ;  
+}
+
+void SpecialPointCalculation :: 
+CalcSpecialPoints (const CSGeometry & ageometry, 
+		   ARRAY<MeshPoint> & apoints)
+{
+  int i;
+
+  geometry = &ageometry;
+  points = &apoints;
+
+  size = geometry->MaxSize();  // globflags.GetNumFlag ("maxsize", 500);
+  (*testout) << "Find Special Points" << endl;
+  (*testout) << "maxsize = " << size << endl;
+
+  cpeps1 = 1e-6; 
+  epeps1 = 1e-3; 
+  epeps2 = 1e-6; 
+
+  epspointdist2 = sqr (size * 1e-8); 
+  relydegtest = size * 1e-4; 
+
+
+  BoxSphere<3> box (Point<3> (-size, -size, -size),
+		    Point<3> ( size,  size,  size));
+  box.CalcDiamCenter();
+  PrintMessage (3, "main-solids: ", geometry->GetNTopLevelObjects());
+
+  for (i = 0; i < geometry->GetNTopLevelObjects(); i++)
+    {
+      (*testout) << "tlo " << i << ":" << endl;
+      const TopLevelObject * tlo = geometry->GetTopLevelObject(i);
+      tlo->GetSolid()->Print (*testout); 
+      (*testout) << endl;
+      CalcSpecialPointsRec (tlo->GetSolid(), tlo->GetLayer(),
+			    box, 1, 1, 1);
+    }
+  
+  PrintDot ('\n');
+
+
+  // add user point:
+  int found = 0;
+  for (i = 0; i < geometry->GetNUserPoints(); i++)
+    AddPoint (geometry->GetUserPoint(i), 1);
+  
+  PrintMessage (3, apoints.Size(), " special points");
+
+  for (i = 0; i < boxesinlevel.Size(); i++)
+    (*testout) << "level " << i << " has " 
+	       << boxesinlevel[i] << " boxes" << endl;
+}
+
+
+
+int debug;
+void SpecialPointCalculation :: 
+CalcSpecialPointsRec (const Solid * sol, int layer,
+		      const BoxSphere<3> & box, 
+		      int level, bool calccp, bool calcep)
+{
+  if (multithread.terminate)
+    return;
+
+  int i;
+  BoxSphere<3> sbox;
+  Solid * redsol;
+
+  int numprim;
+
+  bool decision;
+  bool possiblecrossp, possibleexp;  // possible cross or extremalpoint
+  bool surecrossp, sureexp;          // sure ...
+  
+  static ARRAY<int> locsurf;  // attention: array is static
+
+
+  Point<3> p;
+  int k1, k2, k3;
+  int extremdir;
+  double hd;
+
+  if (!sol) return;
+
+
+  if (level >= 100)
+    {
+      cerr << "Problems in CalcSpecialPoints" << endl;
+      cerr << "Point: " << box.Center() << endl;
+      exit (1);
+    }
+
+  static int cntbox = 0;
+  cntbox++;
+  if (cntbox % 10000 == 0) 
+    PrintDot ();
+
+  if (level <= boxesinlevel.Size())
+    boxesinlevel.Elem(level)++;
+  else
+    boxesinlevel.Append (1);
+
+  /*
+  numprim = sol -> NumPrimitives();
+  sol -> GetSurfaceIndices (locsurf);
+  */
+
+  debug = 0;
+  // box.IsIn (Point<3> (4.9, 1.279, 2.8));
+
+
+  geometry -> GetIndependentSurfaceIndices (sol, box, locsurf);
+  numprim = locsurf.Size();
+  
+  if (debug)
+    {
+      (*testout) << "box = " << box.PMin() << "-" << box.PMax()
+		 << " np = " << numprim << " : ";
+      for (i = 1; i <= locsurf.Size(); i++)
+	(*testout) << " " << locsurf.Get(i);
+      (*testout) << " diam = " << box.Diam();
+      (*testout) << " numprim = " << numprim;
+      (*testout) << endl;
+    }
+
+  p = box.Center();
+  
+  /*
+  (*testout) << "box = " << box.PMin() << " - " << box.PMax()
+	     << ", lev = " << level 
+	     << ", nprim = " << sol->NumPrimitives() 
+	     << ", lsurf = " << locsurf.Size() << endl;
+
+  for (i = 1; i <= locsurf.Size(); i++)
+    geometry->GetSurface (locsurf.Get(i)) -> Print (*testout);
+  sol -> Print (*testout);
+  */
+
+  /*
+    for (i = 1; i <= locsurf.Size(); i++)
+    (*testout) << locsurf.Get(i) << " ";
+    (*testout) << "C = " << box.Center() << " diam = " << box.Diam() << endl;
+    */
+
+  possiblecrossp = (numprim >= 3) && calccp;
+  surecrossp = 0;
+
+  if (possiblecrossp && (locsurf.Size() <= 10))
+    {
+      decision = 1;
+      surecrossp = 0;
+
+      for (k1 = 1; k1 <= locsurf.Size() - 2; k1++)
+	for (k2 = k1 + 1; k2 <= locsurf.Size() - 1; k2++)
+	  for (k3 = k2 + 1; k3 <= locsurf.Size(); k3++)
+	    {
+	      int nc, deg;
+	      nc = CrossPointNewtonConvergence 
+		(geometry->GetSurface(locsurf.Get(k1)), 
+		 geometry->GetSurface(locsurf.Get(k2)), 
+		 geometry->GetSurface(locsurf.Get(k3)), p );
+	      
+	      deg = CrossPointDegenerated 
+		(geometry->GetSurface(locsurf.Get(k1)), 
+		 geometry->GetSurface(locsurf.Get(k2)), 
+		 geometry->GetSurface(locsurf.Get(k3)), box );
+	      
+	      if (!nc && !deg) decision = 0;
+	      if (nc) surecrossp = 1;
+	    }
+
+      if (decision && surecrossp)
+	{
+	  for (k1 = 1; k1 <= locsurf.Size() - 2; k1++)
+	    for (k2 = k1 + 1; k2 <= locsurf.Size() - 1; k2++)
+	      for (k3 = k2 + 1; k3 <= locsurf.Size(); k3++)
+		{
+		  if (CrossPointNewtonConvergence 
+		      (geometry->GetSurface(locsurf.Get(k1)), 
+		       geometry->GetSurface(locsurf.Get(k2)), 
+		       geometry->GetSurface(locsurf.Get(k3)), p ) )
+		    {
+		      Point<3> pp = p;
+		      CrossPointNewton 
+			(geometry->GetSurface(locsurf.Get(k1)), 
+			 geometry->GetSurface(locsurf.Get(k2)), 
+			 geometry->GetSurface(locsurf.Get(k3)), pp);
+              
+		      BoxSphere<3> hbox (pp, pp);
+		      hbox.Increase (1e-8);
+
+		      if (pp(0) > box.PMin()(0) - 1e-5 && 
+			  pp(0) < box.PMax()(0) + 1e-5 &&
+			  pp(1) > box.PMin()(1) - 1e-5 && 
+			  pp(1) < box.PMax()(1) + 1e-5 &&
+			  pp(2) > box.PMin()(2) - 1e-5 && 
+			  pp(2) < box.PMax()(2) + 1e-5 &&
+			  sol -> IsIn (pp) && !sol->IsStrictIn (pp) &&
+			  !CrossPointDegenerated
+			  (geometry->GetSurface(locsurf.Get(k1)), 
+			   geometry->GetSurface(locsurf.Get(k2)), 
+			   geometry->GetSurface(locsurf.Get(k3)), hbox ))
+
+			{ 
+			  //                AddCrossPoint (locsurf, sol, p);
+			  BoxSphere<3> boxp (pp, pp);
+			  boxp.Increase (1e-3);
+			  boxp.CalcDiamCenter();
+			  ARRAY<int> locsurf2;
+
+			  geometry -> GetIndependentSurfaceIndices (sol, boxp, locsurf2);
+			  
+			  /*
+			  ReducePrimitiveIterator rpi(boxp);
+			  UnReducePrimitiveIterator urpi;
+			  
+			  ((Solid*)sol) -> IterateSolid (rpi);
+			  sol -> GetIndependentSurfaceIndices (locsurf2);
+			  ((Solid*)sol) -> IterateSolid (urpi);
+			  */
+			  bool found1 = 0, found2 = 0, found3 = 0;
+			  for (i = 1; i <= locsurf2.Size(); i++)
+			    {
+			      if (locsurf2.Get(i) == locsurf.Get(k1))
+				found1 = 1;
+			      if (locsurf2.Get(i) == locsurf.Get(k2))
+				found2 = 1;
+			      if (locsurf2.Get(i) == locsurf.Get(k3))
+				found3 = 1;
+			    }
+			  
+			  if (found1 && found2 && found3)
+			    if (AddPoint (pp, layer))
+			      {
+				(*testout) << "Crosspoint found: " << pp 
+					   << " diam = " << box.Diam() << endl;
+				(*testout) << "surfs: " 
+					   << locsurf.Get(k1) << "," 
+					   << locsurf.Get(k2) << "," 
+					   << locsurf.Get(k3) << endl;
+			      }
+			}
+		    }
+		}
+	}
+      
+      if (decision)
+	possiblecrossp = 0;
+    }
+
+
+
+
+  possibleexp = (numprim >= 2) && calcep;
+    
+  if (possibleexp && (locsurf.Size() <= 10))
+    {
+      decision = 1;
+      sureexp = 0;
+
+      for (k1 = 1; k1 <= locsurf.Size() - 1; k1++)
+	for (k2 = k1+1; k2 <= locsurf.Size(); k2++)
+	  {
+	    bool nc, deg;
+	    
+	    nc = EdgeNewtonConvergence 
+	      (geometry->GetSurface(locsurf.Get(k1)),
+	       geometry->GetSurface(locsurf.Get(k2)),
+	       p);
+	    
+	    deg = EdgeDegenerated 
+	      (geometry->GetSurface(locsurf.Get(k1)),
+	       geometry->GetSurface(locsurf.Get(k2)),
+	       box);
+	    
+	    if (!nc && !deg) decision = 0;
+	    if (nc) sureexp = 1;
+
+	    if (debug)
+	      {
+		(*testout) << "p = " << p << " s1,2 = " << locsurf.Get(k1) << ", " << locsurf.Get(k2) 
+			   << " nc = " << nc << " deg = " << deg << endl;
+	      }
+	  }
+
+      if (decision && sureexp)
+	{
+	  for (k1 = 1; k1 <= locsurf.Size() - 1; k1++)
+	    for (k2 = k1+1; k2 <= locsurf.Size(); k2++)
+	      {
+		if (
+		    EdgeNewtonConvergence 
+		    (geometry->GetSurface(locsurf.Get(k1)),
+		     geometry->GetSurface(locsurf.Get(k2)),
+		     p) )
+		  {
+		    EdgeNewton 
+		      (geometry->GetSurface(locsurf.Get(k1)),
+		       geometry->GetSurface(locsurf.Get(k2)),
+		       p);
+		    
+		    Point<3> pp;
+		    if (IsEdgeExtremalPoint 
+		      (geometry->GetSurface(locsurf.Get(k1)),
+		       geometry->GetSurface(locsurf.Get(k2)),
+		       p, pp, box.Diam()/2))
+			{
+			  (*testout) << "extremalpoint (nearly) found:" 
+				     << pp
+				     << endl;
+			  if (Dist (pp, box.Center()) < box.Diam()/2 &&
+			      sol -> IsIn (pp) && !sol->IsStrictIn (pp) )
+			    {
+			      //                AddExtremalPoint (locsurf.Get(k1), locsurf.Get(k2), p);
+			      if (AddPoint (pp, layer))
+				(*testout) << "Extremal point found: " << pp << endl;
+			    }  
+			}            
+		  }
+	      }
+	}
+      if (decision)
+	possibleexp = 0;
+    }
+ 
+
+
+  if (possiblecrossp || possibleexp)
+    {
+      for (i = 0; i < 8; i++)
+	{
+	  box.GetSubBox (i, sbox);
+	  sbox.Increase (1e-4 * sbox.Diam());
+
+	  redsol = sol -> GetReducedSolid (sbox);
+
+	  if (redsol)
+	    {
+	      CalcSpecialPointsRec (redsol, layer, sbox, level+1, calccp, calcep);
+	      delete redsol;
+	    }
+	}
+    }
+}
+
+
+
+
+
+/******* Tests for Point of intersection **********************/
+
+
+
+bool SpecialPointCalculation :: 
+CrossPointNewtonConvergence (const Surface * f1, 
+			     const Surface * f2, 
+			     const Surface * f3,
+			     const Point<3> & p)
+{
+  int i;
+  Vec<3> grad;
+  Vec<3> rs, x;
+  Mat<3> jacobi, inv;
+  double alpha, beta, gamma, eta;
+  double sum;
+  int j;
+
+
+  rs(0) = f1->CalcFunctionValue (p);
+  rs(1) = f2->CalcFunctionValue (p);
+  rs(2) = f3->CalcFunctionValue (p);
+
+  f1->CalcGradient (p, grad);
+  jacobi(0,0) = grad(0);
+  jacobi(0,1) = grad(1);
+  jacobi(0,2) = grad(2);
+
+  f2->CalcGradient (p, grad);
+  jacobi(1,0) = grad(0);
+  jacobi(1,1) = grad(1);
+  jacobi(1,2) = grad(2);
+
+  f3->CalcGradient (p, grad);
+  jacobi(2,0) = grad(0);
+  jacobi(2,1) = grad(1);
+  jacobi(2,2) = grad(2);
+
+  alpha = 1;
+  if (fabs (Det (jacobi)) > 1e-8)
+    {
+      CalcInverse (jacobi, inv);
+      x = inv * rs;
+
+      gamma = f1 -> HesseNorm() + f2 -> HesseNorm() + f3 -> HesseNorm();
+      beta = 0;
+      for (i = 0; i < 3; i++)
+	{
+	  sum = 0;
+	  for (j = 0; j < 3; j++)
+	    sum += fabs (inv(i,j));
+	  beta = max2 (beta, sum);
+	}
+      eta = Abs (x);
+
+      alpha = beta * gamma * eta;
+    }
+
+  return (alpha < 0.1);
+}
+
+
+
+
+bool SpecialPointCalculation :: 
+CrossPointDegenerated (const Surface * f1,
+		       const Surface * f2, 
+		       const Surface * f3, 
+		       const BoxSphere<3> & box) const
+{
+  Mat<3> mat;
+  Vec<3> grad, g1, g2, g3;
+  double normprod;
+
+  if (box.Diam() > relydegtest) return 0;
+
+  f1->CalcGradient (box.Center(), g1);
+  normprod = Abs (g1);
+
+  f2->CalcGradient (box.Center(), g2);
+  normprod *= Abs (g2);
+ 
+  f3->CalcGradient (box.Center(), g3);
+  normprod *= Abs (g3);
+
+  for (int i = 0; i < 3; i++)
+    {
+      mat(i,0) = g1(i);
+      mat(i,1) = g2(i);
+      mat(i,2) = g3(i);
+    }
+
+  if (fabs (Det (mat)) < cpeps1 * normprod)
+    return 1;
+  else 
+    return 0;
+}
+ 
+
+
+
+
+void SpecialPointCalculation :: CrossPointNewton (const Surface * f1, 
+						  const Surface * f2, 
+						  const Surface * f3, Point<3> & p)
+{
+  int i;
+  Vec<3> g1, g2, g3;
+  Vec<3> rs, sol;
+  Mat<3> mat;
+
+  i = 10;
+  while (i > 0)
+    {
+      i--;
+      rs(0) = f1->CalcFunctionValue (p);
+      rs(1) = f2->CalcFunctionValue (p);
+      rs(2) = f3->CalcFunctionValue (p);
+
+      f1->CalcGradient (p, g1);
+      f2->CalcGradient (p, g2);
+      f3->CalcGradient (p, g3);
+
+      for (int j = 0; j < 3; j++)
+	{
+	  mat(0, j) = g1(j);
+	  mat(1, j) = g2(j);
+	  mat(2, j) = g3(j);
+	}
+      mat.Solve (rs, sol);
+	  /*
+      Transpose (g1, g2, g3);
+      SolveLinearSystem (g1, g2, g3, rs, sol);
+	  */
+      if (sol.Length() < 1e-12 && i > 1) i = 1;
+
+      p -= sol;
+    }
+}
+
+
+
+
+/******* Tests for Point on edges **********************/
+
+
+
+
+bool SpecialPointCalculation :: 
+EdgeNewtonConvergence (const Surface * f1, const Surface * f2, 
+		       const Point<3> & p)
+{
+  int i;
+  Vec<3> g1, g2, sol;
+  Vec<2> vrs;
+  double alpha, beta, gamma, eta;
+  double sum;
+  Mat<2,3> mat;
+  Mat<3,2> inv;
+  int j;
+
+  vrs(0) = f1->CalcFunctionValue (p);
+  vrs(1) = f2->CalcFunctionValue (p);
+
+  f1->CalcGradient (p, g1);
+  f2->CalcGradient (p, g2);
+
+  for (i = 0; i < 3; i++)
+    {
+      mat(0,i) = g1(i);
+      mat(1,i) = g2(i);
+    }
+
+  alpha = 1;
+
+  if ( fabs(g1 * g2) < (1 - 1e-8) * Abs (g1) * Abs (g2))
+    {
+      CalcInverse (mat, inv);
+      sol = inv * vrs;
+
+      // SolveLinearSystemLS (g1, g2, vrs, sol);
+
+      gamma = f1 -> HesseNorm() + f2 -> HesseNorm();
+
+      /*
+      Vec<3> inv1, inv2;
+      PseudoInverse (g1, g2, inv1, inv2);
+      */
+      
+      beta = 0;
+      for (i = 0; i < 3; i++)
+	for (j = 0; j < 2; j++)
+	  beta += inv(i,j) * inv(i,j);
+      beta = sqrt (beta);
+
+      //      beta = inv1.Length() + inv2.Length();
+      eta = Abs (sol);
+      alpha = beta * gamma * eta;
+    }
+  return (alpha < 0.1);
+}
+
+
+
+
+bool SpecialPointCalculation :: 
+EdgeDegenerated (const Surface * f1,
+		 const Surface * f2, 
+		 const BoxSphere<3> & box) const
+{
+  // perform newton steps. normals parallel ?
+  // if not decideable: return 0 
+  
+  
+  Point<3> p = box.Center();
+  int i;
+  Vec<3> grad, g1, g2, sol;
+  Vec<2> vrs;
+  Mat<2,3> mat;
+
+  i = 20;
+  while (i > 0)
+    {
+      if (Dist (p, box.Center()) > box.Diam())
+	return 0;
+
+      i--;
+      vrs(0) = f1->CalcFunctionValue (p);
+      vrs(1) = f2->CalcFunctionValue (p);
+
+      f1->CalcGradient (p, g1);
+      f2->CalcGradient (p, g2);
+
+      if ( fabs (g1 * g2) > (1 - 1e-10) * Abs (g1) * Abs (g2))
+	return 1;
+
+      for (int j = 0; j < 3; j++)
+	{
+	  mat(0,j) = g1(j);
+	  mat(1,j) = g2(j);
+	}
+      mat.Solve (vrs, sol);
+      //      SolveLinearSystemLS (g1, g2, vrs, sol);
+
+      if (Abs (sol) < 1e-12 && i > 1) i = 1;
+      p -= sol;
+    }
+
+  return 0;
+  /*
+  return 0;
+
+  static DenseMatrix jacobi(3);
+  Vec<3> grad, g1, g2, g3;
+  double normprod;
+  
+  if (box.Diam() > relydegtest) return 0;
+  
+  f1->CalcGradient (box.Center(), g1);
+  normprod = g1.Length();
+  
+  f2->CalcGradient (box.Center(), g2);
+  normprod *= g2.Length();
+  
+  if (fabs (g1 * g2) < 1e-8 * normprod)
+    return 1;
+  else 
+    return 0;
+  */
+}
+
+
+
+
+
+
+void SpecialPointCalculation :: EdgeNewton (const Surface * f1, 
+					    const Surface * f2, Point<3> & p)
+{
+  int i;
+  Vec<3> grad, g1, g2, sol;
+  Vec<2> vrs;
+  Mat<2,3> mat;
+
+  i = 10;
+  while (i > 0)
+    {
+      i--;
+      vrs(0) = f1->CalcFunctionValue (p);
+      vrs(1) = f2->CalcFunctionValue (p);
+
+      f1->CalcGradient (p, g1);
+      f2->CalcGradient (p, g2);
+
+      for (int j = 0; j < 3; j++)
+	{
+	  mat(0,j) = g1(j);
+	  mat(1,j) = g2(j);
+	}
+      mat.Solve (vrs, sol);
+      //      SolveLinearSystemLS (g1, g2, vrs, sol);
+
+      if (Abs (sol) < 1e-12 && i > 1) i = 1;
+      p -= sol;
+    }
+}
+
+
+
+bool SpecialPointCalculation :: 
+IsEdgeExtremalPoint (const Surface * f1, const Surface * f2, 
+		     const Point<3> & p, Point<3> & pp, double rad)
+{
+  Vec<3> g1, g2, t, t1, t2;
+  int j;
+
+  f1->CalcGradient (p, g1);
+  f2->CalcGradient (p, g2);
+  
+  t = Cross (g1, g2);
+  t.Normalize();
+
+  Point<3> p1 = p + rad * t;
+  Point<3> p2 = p - rad * t;
+
+  EdgeNewton (f1, f2, p1);
+  EdgeNewton (f1, f2, p2);
+
+  
+  f1->CalcGradient (p1, g1);
+  f2->CalcGradient (p1, g2);
+  t1 = Cross (g1, g2);
+  t1.Normalize();
+
+  f1->CalcGradient (p2, g1);
+  f2->CalcGradient (p2, g2);
+  t2 = Cross (g1, g2);
+  t2.Normalize();
+
+  double val = 1e-8 * rad * rad;
+  for (j = 0; j < 3; j++)
+    if ( (t1(j) * t2(j) < -val) )
+      {
+	pp = p;
+	ExtremalPointNewton (f1, f2, j+1, pp);
+	return 1;
+      }
+
+  return 0;
+}
+
+
+
+
+
+
+
+
+
+/********** Tests of Points of extremal coordinates  ****************/
+
+
+void SpecialPointCalculation :: ExtremalPointNewton (const Surface * f1, 
+						     const Surface * f2, 
+						     int dir, Point<3> & p)
+{
+  int i;
+
+  Vec<3> g1, g2, v, curv;
+  Vec<3> rs, x, y1, y2, y;
+  Mat<3> h1, h2;
+  Mat<3> jacobi;
+
+
+  if (dir < 1 || dir > 3)
+    {
+      cerr << "Error: Illegal extremdir" << endl;
+      return;
+    }
+
+  i = 50;
+  while (i > 0)
+    {
+      i--;
+      rs(0) = f1->CalcFunctionValue (p);
+      rs(1) = f2->CalcFunctionValue (p);
+
+      f1 -> CalcGradient (p, g1);
+      f2 -> CalcGradient (p, g2);
+
+      f1 -> CalcHesse (p, h1);
+      f2 -> CalcHesse (p, h2);
+
+
+      v = Cross (g1, g2);
+
+      rs(2) = v(dir-1);
+
+      jacobi(0,0) = g1(0);
+      jacobi(0,1) = g1(1);
+      jacobi(0,2) = g1(2);
+
+      jacobi(1,0) = g2(0);
+      jacobi(1,1) = g2(1);
+      jacobi(1,2) = g2(2);
+
+
+      switch (dir)
+	{
+	case 1:
+	  {
+	    y1(0) = 0;
+	    y1(1) = g2(2);
+	    y1(2) = -g2(1);
+	    y2(0) = 0;
+	    y2(1) = -g1(2);
+	    y2(2) = g1(1);
+	    break;
+	  }
+	case 2:
+	  {
+	    y1(0) = -g2(2);
+	    y1(1) = 0;
+	    y1(2) = g2(0);
+	    y2(0) = g1(2);
+	    y2(1) = 0;
+	    y2(2) = -g1(0);
+	    break;
+	  }
+	case 3:
+	  {
+	    y1(0) = g2(1);
+	    y1(1) = -g2(0);
+	    y1(2) = 0;
+	    y2(0) = -g1(1);
+	    y2(1) = g1(0);
+	    y2(2) = 0;
+	    break;
+	  }
+	}
+
+      y = h1 * y1 + h2 * y2;
+
+      jacobi(2,0) = y(0);
+      jacobi(2,1) = y(1);
+      jacobi(2,2) = y(2);
+
+      jacobi.Solve (rs, x);
+      /*
+      CalcInverse (jacobi, inv);
+      inv.Mult (rs, x);
+      */
+      //    (*testout) << "err = " << x.L2Norm() << endl;
+
+      if (Abs (x) < 1e-12 && i > 1)
+	{
+	  //      (*testout) << "convergent in " << (10 - i) << " steps " << endl;
+
+	  i = 1;
+	}
+      
+      p -= x;
+    }
+
+  if (Abs (x) > 1e-10)
+    {
+      (*testout) << "Error: extremum Newton not convergent" << endl;
+      (*testout) << "dir = " << dir << endl;
+      (*testout) << "p = " << p << endl;
+      (*testout) << "x = " << x << endl;
+    }
+}
+
+
+
+
+bool SpecialPointCalculation :: ExtremalPointPossible (const Surface * f1, 
+						       const Surface * f2, 
+						       int dir, 
+						       const BoxSphere<3> & box)
+{
+  double hn1, hn2, gn1, gn2;
+  Point<3> p;
+  Vec<3> g1, g2, v;
+  double f3;
+  double r = box.Diam()/2;
+
+  p = box.Center();
+
+  f1 -> CalcGradient (p, g1);
+  f2 -> CalcGradient (p, g2);
+
+  gn1 = g1.Length();
+  gn2 = g2.Length();
+
+  hn1 = f1 -> HesseNorm ();
+  hn2 = f2 -> HesseNorm ();
+
+  v = Cross (g1, g2);
+  f3 = fabs (v(dir-1));
+
+  //  (*testout) << "f3 = " << f3 << "  r = " << r 
+  //             << "normbound = " 
+  //             << (hn1 * (gn2 + r * hn2) + hn2 * (gn1 + r * hn1)) << endl;
+ 
+  return (f3 <= 3 * r * (hn1 * (gn2 + r * hn2) + hn2 * (gn1 + r * hn1)));
+}
+
+
+
+bool SpecialPointCalculation :: 
+ExtremalPointNewtonConvergence (const Surface * f1, const Surface * f2, 
+				int dir, 
+				const BoxSphere<3> & box)
+{
+  return box.Diam() < 1e-8;
+}
+
+
+bool SpecialPointCalculation :: 
+ExtremalPointDegenerated (const Surface * f1, const Surface * f2, 
+			  int dir, const BoxSphere<3> & box)
+{
+  double gn1, gn2;
+  Point<3> p;
+  Vec<3> g1, g2, v;
+  double maxderiv;
+  double minv;
+  Vec<3> curv, t;
+  Vec<2> rs, x;
+  Mat<3> h1, h2;
+  Mat<2> a, inv;
+  double leftside;
+
+  if (box.Diam() > relydegtest) return 0;
+
+  p = box.Center();
+
+  f1 -> CalcGradient (p, g1);
+  f2 -> CalcGradient (p, g2);
+  gn1 = g1.Length();
+  gn2 = g2.Length();
+
+  v = Cross (g1, g2);
+  if (Abs (v) < epeps1 * gn1 * gn2) return 1;       // irregular edge
+
+  f1 -> CalcHesse (p, h1);
+  f2 -> CalcHesse (p, h2);
+
+  //  hn1 = f1 -> HesseNorm ();
+  //  hn2 = f2 -> HesseNorm ();
+
+  t = v;
+  a(0, 0) = g1 * g1;
+  a(0, 1) = 
+    a(1, 0) = g1 * g2;
+  a(1, 1) = g2 * g2;
+  
+  rs(0) = g1(dir-1);
+  rs(1) = g2(dir-1);
+
+  a.Solve (rs, x);
+
+  /*
+  CalcInverse (a, inv);
+  inv.Mult (rs, x);          // x .. Lagrangeparameter
+  */
+  //  (*testout) << "g1 = " << g1 << " g2 = " << g2 << endl;
+  //  (*testout) << "lam = " << x << endl;
+  //  (*testout) << "h2 = " << h2 << endl;
+
+  leftside = fabs (x(0) * ( t * (h1 * t)) + 
+                   x(1) * ( t * (h2 * t)));
+
+  //  (*testout) << "leftside = " << leftside << endl;
+
+  if (leftside < epeps2 * Abs2 (v)) return 1;  
+
+  return 0;
+}
+
+ 
+
+bool SpecialPointCalculation :: AddPoint (const Point<3> & p, int layer)
+{
+  for (int i = 0; i < points->Size(); i++)
+    if (Dist2 ( (*points)[i], p) < epspointdist2 &&
+	(*points)[i].GetLayer() == layer)
+      return 0;
+
+  points->Append (MeshPoint(p, layer));
+  return 1;
+}
+
+
+
+
+
+
+
+/*
+void SpecialPointCalculation :: 
+AnalyzeSpecialPoints (const CSGeometry & ageometry,
+		      ARRAY<Point<3> > & apoints, 
+		      ARRAY<SpecialPoint> & specpoints)
+{
+  int si, i, j, k, l, m, spi;
+  Solid * locsol;
+  ARRAY<int> surfind;
+  ARRAY<Vec<3>> normalvecs;
+  const Solid * sol;
+  Vec<3> t;
+  Point<3> p;
+
+  ARRAY<int> specpoint2point;
+  specpoints.SetSize (0);
+ 
+  (*testout) << "AnalyzeSpecialPoints" << endl;
+
+  for (si = 1; si <= ageometry.GetNTopLevelObjects(); si++)
+    {
+      (*testout) << "main solid " << si << endl;
+
+      sol = ageometry.GetTopLevelObject(si)->GetSolid();
+      for (i = 1; i <= apoints.Size(); i++)
+	{
+	  p = apoints.Get(i);
+
+	  sol -> TangentialSolid (p, locsol);
+	  if (!locsol) continue;
+	  
+	  (*testout) << "Point " << apoints.Get(i) << endl;
+
+	  locsol -> GetSurfaceIndices (surfind);
+	  for (j = surfind.Size(); j >= 1; j--)
+	    if (fabs (ageometry.GetSurface(surfind.Get(j))->
+		      CalcFunctionValue (p)) > 1e-6)
+	      surfind.DeleteElement (j);
+
+
+
+	  (*testout) << "Surfaces: ";
+	  for (j = 1; j <= surfind.Size(); j++)
+	    (*testout) << surfind.Get(j) << " ";
+	  (*testout) << endl;
+
+	  normalvecs.SetSize(surfind.Size());
+	  for (j = 1; j <= surfind.Size(); j++)
+	    ageometry.GetSurface(surfind.Get(j)) -> 
+	      GetNormalVector(apoints.Get(i), normalvecs.Elem(j));
+
+	  for (j = 1; j <= normalvecs.Size() - 1; j ++)
+	    for (k = j+1; k <= normalvecs.Size(); k++)
+	      for (l = 1; l <= 2; l++)
+		{
+		  t = Cross (normalvecs.Get(j), normalvecs.Get(k));
+		  if (t.Length2() < 1e-8)
+		    {
+		      cerr << "AnalyzePoint: Surfaces degenerated" << endl;
+		      break;
+		    }
+		  t /= t.Length();
+		  if (l == 2) t *= -1;
+
+		  if (locsol -> Edge (apoints.Get(i), t))
+		    {
+		      spi = 0;
+		      for (m = 1; m <= specpoints.Size(); m++)
+			if (Dist2 (specpoints.Get(m).p, apoints.Get(i)) < 1e-8
+			    && (specpoints.Get(m).v - t).Length2() < 1e-8)
+			  {
+			    spi = m;
+			    break;
+			  }
+		      if (!spi)
+			{
+			  spi = specpoints.Append (SpecialPoint());
+			  specpoint2point.Append (i);
+			  specpoints.Last().unconditional = 0;
+			}
+		      specpoints.Elem(spi).p = apoints.Get(i);
+		      specpoints.Elem(spi).v = t;
+		      if (surfind.Size() >= 3)
+			specpoints.Elem(spi).unconditional = 1;
+		      specpoints.Elem(spi).s1 = surfind.Get(j);
+		      specpoints.Elem(spi).s2 = surfind.Get(k);
+		      (*testout) << "spi = " << spi 
+				 << " uncond = " << specpoints.Get(spi).unconditional
+				 << " t = " << t << endl;
+		    }
+          
+		}
+	  delete locsol;
+	}
+    }
+
+  // if special point is unconditional on some solid,
+  // it must be unconditional everywhere:
+
+  BitArray uncond (apoints.Size());
+  uncond.Clear();
+
+  for (i = 1; i <= specpoints.Size(); i++)
+    if (specpoints.Get(i).unconditional)
+      uncond.Set (specpoint2point.Get(i));
+
+  for (i = 1; i <= specpoints.Size(); i++)
+    specpoints.Elem(i).unconditional = 
+      uncond.Test (specpoint2point.Get(i)) ? 1 : 0;
+}
+*/
+
+
+
+void SpecialPointCalculation :: 
+AnalyzeSpecialPoints (const CSGeometry & ageometry,
+		      ARRAY<MeshPoint> & apoints, 
+		      ARRAY<SpecialPoint> & specpoints)
+{
+  int si, i, j, k, l, m, spi;
+
+  Solid * locsol;
+  ARRAY<int> surfind;
+  ARRAY<int> surfind2;
+
+  ARRAY<Vec<3> > normalvecs;
+  const Solid * sol;
+  const Surface * surf;
+
+  Vec<3> t, nsurf;
+  Point<3> p;
+
+  ARRAY<int> specpoint2point;
+  specpoints.SetSize (0);
+
+  geometry = &ageometry;
+ 
+  (*testout) << "AnalyzeSpecialPoints\n";
+
+  for (si = 0; si < ageometry.GetNTopLevelObjects(); si++)
+    {
+      (*testout) << "main solid " << si << "\n";
+
+      sol = ageometry.GetTopLevelObject(si)->GetSolid();
+      surf = ageometry.GetTopLevelObject(si)->GetSurface();
+
+      for (i = 0; i < apoints.Size(); i++)
+	{
+	  p = apoints[i];
+	  if (ageometry.GetTopLevelObject(si)->GetLayer() !=
+	      apoints[i].GetLayer())
+	    continue;
+
+	  // (*testout) << "Point " << apoints[i] << "\n";
+
+	  sol -> TangentialSolid (p, locsol);
+	  if (!locsol) continue;
+	  
+	  // get all surface indices, 
+	  if (surf)
+	    {
+	      locsol -> GetSurfaceIndices (surfind);
+	      bool hassurf = 0;
+	      for (m = 0; m < surfind.Size(); m++)
+		if (ageometry.GetSurface(surfind[m]) == surf)
+		  hassurf = 1;
+
+	      if (!hassurf)
+		continue;
+
+	      surf->GetNormalVector (p, nsurf);
+	    }
+
+	  // get independent surfaces of tangential solid
+
+	  BoxSphere<3> box(p,p);
+	  box.Increase (1e-6);
+	  box.CalcDiamCenter();
+	  ageometry.GetIndependentSurfaceIndices (locsol, box, surfind);
+
+
+	  /*
+	  locsol -> GetSurfaceIndices (surfind);
+	  for (j = surfind.Size(); j >= 1; j--)
+	    if (fabs (ageometry.GetSurface(surfind.Get(j))->
+		      CalcFunctionValue (p)) > 1e-6)
+	      surfind.DeleteElement (j);
+	  */
+
+	  /*
+	  (*testout) << "Surfaces: ";
+	  for (j = 0; j < surfind.Size(); j++)
+	    (*testout) << surfind[j] << " ";
+	  (*testout) << "\n";
+	  */
+
+
+	  normalvecs.SetSize(surfind.Size());
+	  for (j = 0; j < surfind.Size(); j++)
+	    ageometry.GetSurface(surfind[j]) ->
+	      GetNormalVector(apoints[i], normalvecs[j]);
+
+	  for (j = 0; j < normalvecs.Size(); j++)
+	    for (k = j+1; k < normalvecs.Size(); k++)
+	      for (l = 1; l <= 2; l++)
+		{
+		  t = Cross (normalvecs[j], normalvecs[k]);
+		  if (Abs2 (t) < 1e-8)
+		    {
+		      cerr << "AnalyzePoint: Surfaces degenerated" << "\n";
+		      break;
+		    }
+		  t.Normalize();
+		  if (l == 2) t *= -1;
+
+		  // try tangential direction t
+
+		  // (*testout) << "check tangential " << t << "\n";
+
+		  if (surf && fabs (nsurf * t) > 1e-6)
+		    continue;
+
+		  if (!surf)
+		    {
+		      ageometry.GetIndependentSurfaceIndices 
+			(locsol, p, t, surfind2);
+		  
+		      bool found1 = 0, found2 = 0;
+		      for (int ii = 0; ii < surfind2.Size(); ii++)
+			{
+			  if (surfind2[ii] == surfind[j])
+			    found1 = 1;
+			  if (surfind2[ii] == surfind[k])
+			    found2 = 1;
+			}
+		      if (!found1 || !found2)
+			continue;
+		    }
+
+
+		  bool isedge;
+
+		  // isedge = locsol -> Edge (apoints.Get(i), t);
+		  
+		  // edge must be on tangential surface
+		  isedge = 
+		    locsol->VectorIn (p, t) &&
+		    !locsol->VectorStrictIn (p, t);
+		  
+		  // (*testout) << "isedge,1 = " << isedge << "\n";
+
+		  // there must exist at least two different faces on edge
+		  if (isedge)
+		    {
+		      int cnts = 0;
+		      for (m = 0; m < surfind.Size(); m++)
+			{
+			  if (fabs (normalvecs[m] * t) > 1e-6)
+			    continue;
+
+			  Vec<3> s = Cross (normalvecs[m], t);
+			  Vec<3> t2a = t + 0.01 *s;
+			  Vec<3> t2b = t - 0.01 *s;
+
+			  /*
+			  (*testout) << "nv = " << normalvecs[m] << ", s = " << s << "\n";
+			  (*testout) << "t2a = " << t2a << ", t2b = " << t2b << "\n";
+			  (*testout) << "via = "
+				     << locsol->VectorIn (p, t2a) << "/"
+				     << locsol->VectorStrictIn (p, t2a);
+			  (*testout) << "vib = "
+				     << locsol->VectorIn (p, t2b) << "/"
+				     << locsol->VectorStrictIn (p, t2b) << "\n";
+			  */
+
+			  bool isface =
+			    (locsol->VectorIn (p, t2a) &&
+			     !locsol->VectorStrictIn (p, t2a))
+			    ||
+			    (locsol->VectorIn (p, t2b) &&
+			     !locsol->VectorStrictIn (p, t2b));
+			  
+			  if (isface)
+			    {
+			      cnts++;
+			    }
+			}
+		      if (cnts < 2) isedge = 0;
+		    }
+
+		  if (isedge)
+		    {
+		      spi = -1;
+		      for (m = 0; m < specpoints.Size(); m++)
+			if (Dist2 (specpoints[m].p, apoints[i]) < 1e-8
+			    && Abs2(specpoints[m].v - t) < 1e-8)
+			  {
+			    spi = m;
+			    break;
+			  }
+		      if (spi == -1)
+			{
+			  spi = specpoints.Append (SpecialPoint()) - 1;
+			  specpoint2point.Append (i);
+			  specpoints.Last().unconditional = 0;
+			}
+		      specpoints[spi].p = apoints[i];
+		      specpoints[spi].v = t;
+		      if (surfind.Size() >= 3)
+			specpoints[spi].unconditional = 1;
+		      specpoints[spi].s1 = surfind[j];
+		      specpoints[spi].s2 = surfind[k];
+		      specpoints[spi].layer = apoints[i].GetLayer();
+		      for (int up = 0; up < geometry->GetNUserPoints(); up++)
+			if (Dist (geometry->GetUserPoint(up), apoints[i]) < 1e-10)
+			  specpoints[spi].unconditional = 1;
+			
+		      /*
+		      (*testout) << "spi = " << spi 
+				 << " uncond = " << specpoints[spi].unconditional
+				 << " t = " << t << "\n";
+		      */
+		    }
+          
+		}
+	  delete locsol;
+	}
+    }
+
+  // if special point is unconditional on some solid,
+  // it must be unconditional everywhere:
+
+  BitArray uncond (apoints.Size());
+  uncond.Clear();
+
+  for (i = 0; i < specpoints.Size(); i++)
+    if (specpoints[i].unconditional)
+      uncond.Set (specpoint2point[i]);
+  
+  for (i = 0; i < specpoints.Size(); i++)
+    specpoints[i].unconditional = 
+      uncond.Test (specpoint2point[i]) ? 1 : 0;
+}
+}
diff --git a/Netgen/libsrc/csg/surface.cpp b/Netgen/libsrc/csg/surface.cpp
index 2fd4064562..8b8d58127a 100644
--- a/Netgen/libsrc/csg/surface.cpp
+++ b/Netgen/libsrc/csg/surface.cpp
@@ -61,11 +61,19 @@ void Surface :: CalcHesse (const Point<3> & point, Mat<3> & hesse) const
     }
 }
   
+/*
 void Surface :: GetNormalVector (const Point<3> & p, Vec<3> & n) const
 {
   CalcGradient (p, n);
   n.Normalize();
-  //  if (Inverse()) n *= -1;
+}
+*/
+Vec<3> Surface :: GetNormalVector (const Point<3> & p) const
+{
+  Vec<3> n;
+  CalcGradient (p, n);
+  n.Normalize();
+  return n;
 }
 
 void Surface :: DefineTangentialPlane (const Point<3> & ap1, 
@@ -74,7 +82,7 @@ void Surface :: DefineTangentialPlane (const Point<3> & ap1,
   p1 = ap1;
   p2 = ap2;
   
-  GetNormalVector (p1, ez);
+  ez = GetNormalVector (p1);
   ex = p2 - p1;
   ex -= (ex * ez) * ez;
   ex.Normalize();
@@ -86,7 +94,7 @@ void Surface :: ToPlane (const Point<3> & p3d, Point<2> & pplane,
 {
   Vec<3> p1p, n;
 
-  GetNormalVector (p3d, n);
+  n = GetNormalVector (p3d);
   if (n * ez < 0)
     {
       zone = -1;
@@ -254,7 +262,11 @@ VecInSolid2 (const Point<3> & p,
 	     double eps) const
 {
   Point<3> hp = p + 1e-3 * v1 + 1e-5 * v2;
-  return PointInSolid (hp, eps);
+
+  INSOLID_TYPE res = PointInSolid (hp, eps);
+  //  (*testout) << "vectorin2, type = " << typeid(*this).name() << ", res = " << res << endl;
+
+  return res;
 }
 
 
diff --git a/Netgen/libsrc/csg/surface.hpp b/Netgen/libsrc/csg/surface.hpp
index 2738d5222d..db0b4a74d2 100644
--- a/Netgen/libsrc/csg/surface.hpp
+++ b/Netgen/libsrc/csg/surface.hpp
@@ -90,8 +90,8 @@ public:
   virtual void Project (Point<3> & p) const;
 
 
-  virtual int IsIdentic (const Surface & s2, int & inv, 
-			 double eps) const
+  virtual int IsIdentic (const Surface & /* s2 */, int & /* inv */, 
+			 double /* eps */) const
   { return 0; }
   
   ///
@@ -120,7 +120,8 @@ public:
   /**
     Returns outer normal vector.
    */
-  virtual void GetNormalVector (const Point<3> & p, Vec<3> & n) const;
+  // virtual void GetNormalVector (const Point<3> & p, Vec<3> & n) const;
+  virtual Vec<3> GetNormalVector (const Point<3> & p) const;
 
   /**
     Upper bound for spectral norm of Hesse-matrix
@@ -181,9 +182,9 @@ public:
      Gets Approximation by triangles,
      where qual is about the number of triangles per radius
   */
-  virtual void GetTriangleApproximation (TriangleApproximation & tas, 
-					 const Box<3> & boundingbox, 
-					 double facets) const { };
+  virtual void GetTriangleApproximation (TriangleApproximation & /* tas */, 
+					 const Box<3> & /* boundingbox */, 
+					 double /* facets */ ) const { };
 
 #ifdef MYGRAPH  
   ///
diff --git a/Netgen/libsrc/csg/triapprox.cpp b/Netgen/libsrc/csg/triapprox.cpp
index b314855dae..403716155b 100644
--- a/Netgen/libsrc/csg/triapprox.cpp
+++ b/Netgen/libsrc/csg/triapprox.cpp
@@ -8,55 +8,52 @@
 namespace netgen
 {
 
-TriangleApproximation :: TriangleApproximation ()
-{
-  ;
-}
-
-int TriangleApproximation :: 
-AddTriangle (const TATriangle & tri, bool invert)
-{ 
-  trias.Append (tri);
-  if (invert)
-    {
-      trias.Last()[1] = tri[2];
-      trias.Last()[2] = tri[1];
-    }
-  return trias.Size()-1;
-}
-
-
-void TriangleApproximation :: RemoveUnusedPoints ()
-{
-  BitArray used(GetNP());
-  ARRAY<int> map (GetNP());
-  int i, j;
-  int cnt = 0;
-
-  used.Clear();
-  for (i = 0; i < GetNT(); i++)
-    for (j = 0; j < 3; j++)
-      used.Set (GetTriangle (i)[j]);
-
-  for (i = 0; i < GetNP(); i++)
-    if (used.Test(i))
+  TriangleApproximation :: TriangleApproximation ()
+  {
+    ;
+  }
+
+  int TriangleApproximation :: 
+  AddTriangle (const TATriangle & tri, bool invert)
+  { 
+    trigs.Append (tri);
+    if (invert)
       {
-	map[i] = cnt;
-	cnt++;
+	trigs.Last()[1] = tri[2];
+	trigs.Last()[2] = tri[1];
       }
-  
-  for (i = 0; i < GetNT(); i++)
-    for (j = 0; j < 3; j++)
-      trias[i][j] = map[trias[i][j]];
+    return trigs.Size()-1;
+  }
 
-  for (i = 0; i < GetNP(); i++)
-    if (used.Test(i))
-      {
-	points[map[i]] = points[i];
-	normals[map[i]] = normals[i];
-      }
 
-  points.SetSize (cnt);
-  normals.SetSize (cnt);
-}
+  void TriangleApproximation :: RemoveUnusedPoints ()
+  {
+    BitArray used(GetNP());
+    ARRAY<int> map (GetNP());
+    int i, j;
+    int cnt = 0;
+
+    used.Clear();
+    for (i = 0; i < GetNT(); i++)
+      for (j = 0; j < 3; j++)
+	used.Set (GetTriangle (i)[j]);
+
+    for (i = 0; i < GetNP(); i++)
+      if (used.Test(i))
+	map[i] = cnt++;
+  
+    for (i = 0; i < GetNT(); i++)
+      for (j = 0; j < 3; j++)
+	trigs[i][j] = map[trigs[i][j]];
+
+    for (i = 0; i < GetNP(); i++)
+      if (used.Test(i))
+	{
+	  points[map[i]] = points[i];
+	  normals[map[i]] = normals[i];
+	}
+
+    points.SetSize (cnt);
+    normals.SetSize (cnt);
+  }
 }
diff --git a/Netgen/libsrc/csg/triapprox.hpp b/Netgen/libsrc/csg/triapprox.hpp
index d7192a9a02..7b2db16b09 100644
--- a/Netgen/libsrc/csg/triapprox.hpp
+++ b/Netgen/libsrc/csg/triapprox.hpp
@@ -25,9 +25,6 @@ public:
   int SurfaceIndex() const { return surfind; }
   int & SurfaceIndex() { return surfind; }
 
-  //  int PNum (int i) const { return pi[i-1]; }
-  //  int & PNum (int i) { return pi[i-1]; }
-
   int & operator[] (int i) { return pi[i]; }
   const int & operator[] (int i) const { return pi[i]; }
 };
@@ -37,19 +34,19 @@ class TriangleApproximation
 {
   ARRAY<Point<3> > points;
   ARRAY<Vec<3> > normals;
-  ARRAY<TATriangle> trias;
+  ARRAY<TATriangle> trigs;
 
 public:
   TriangleApproximation();
   int GetNP () const { return points.Size(); }
-  int GetNT () const { return trias.Size(); }
+  int GetNT () const { return trigs.Size(); }
 
   int AddPoint (const Point<3> & p) { points.Append (p); return points.Size()-1; }
   int AddNormal (const Vec<3> & n) { normals.Append (n); return normals.Size()-1; }
   int AddTriangle (const TATriangle & tri, bool invert = 0);
 
   const Point<3> & GetPoint (int i) const { return points[i]; }
-  const TATriangle & GetTriangle (int i) const { return trias[i]; }
+  const TATriangle & GetTriangle (int i) const { return trigs[i]; }
   const Vec<3> & GetNormal (int i) const { return normals[i]; }
 
   void RemoveUnusedPoints ();
diff --git a/Netgen/libsrc/general/array.hpp b/Netgen/libsrc/general/array.hpp
index 020a15ac34..c953cedeff 100644
--- a/Netgen/libsrc/general/array.hpp
+++ b/Netgen/libsrc/general/array.hpp
@@ -27,15 +27,15 @@ protected:
 public:
 
   /// provide size and memory
-  FlatArray (int asize, T * adata) 
+  inline FlatArray (int asize, T * adata) 
     : size(asize), data(adata) { ; }
 
   /// the size
-  int Size() const { return size; }
+  inline int Size() const { return size; }
 
 
   /// access array. 
-  T & operator[] (int i) 
+  inline T & operator[] (int i) 
   { 
 #ifdef DEBUG
     if (i-BASE < 0 || i-BASE >= size)
@@ -47,7 +47,7 @@ public:
 
 
   /// Access array. 
-  const T & operator[] (int i) const
+  inline const T & operator[] (int i) const
   {
 #ifdef DEBUG
     if (i-BASE < 0 || i-BASE >= size)
@@ -215,7 +215,13 @@ public:
   /// Delete element i (0-based). Move last element to position i.
   void Delete (int i)
   {
-    DeleteElement (i+1);
+#ifdef CHECK_ARRAY_RANGE
+    RangeCheck (i+1);
+#endif
+
+    this->data[i] = this->data[this->size-1];
+    this->size--;
+    //    DeleteElement (i+1);
   }
 
 
@@ -297,7 +303,8 @@ template <class T, int S>
 class ArrayMem : public ARRAY<T>
 {
   // T mem[S];
-  char mem[S*sizeof(T)];
+  // char mem[S*sizeof(T)];
+  double mem[(S*sizeof(T)+7) / 8];
 public:
   /// Generate array of logical and physical size asize
   explicit ArrayMem(int asize = 0)
diff --git a/Netgen/libsrc/general/bitarray.hpp b/Netgen/libsrc/general/bitarray.hpp
index 380b78987e..1b7042fa5a 100644
--- a/Netgen/libsrc/general/bitarray.hpp
+++ b/Netgen/libsrc/general/bitarray.hpp
@@ -194,10 +194,10 @@ private:
 template <int BASE>
 inline ostream & operator<< (ostream & s, const BitArrayChar<BASE> & a)
 {
-  for (int i = 0; i < a.Size(); i++)
+  for (int i = BASE; i < a.Size()+BASE; i++)
     {
       s << a.Test(i);
-      if (i % 40 == 0) s << "\n";
+      if ( (i-BASE) % 40 == 39) s << "\n";
     }
   if (a.Size() % 40 != 0) s << "\n";
   return s;
diff --git a/Netgen/libsrc/general/dynamicmem.cpp b/Netgen/libsrc/general/dynamicmem.cpp
index 9cd1f3ad62..66b22cb4a7 100644
--- a/Netgen/libsrc/general/dynamicmem.cpp
+++ b/Netgen/libsrc/general/dynamicmem.cpp
@@ -34,12 +34,12 @@ BaseDynamicMem :: ~BaseDynamicMem ()
   if (prev) prev->next = next;
   else first = next;
 
-  delete name;
+  delete [] name;
 }
 
 void BaseDynamicMem :: SetName (const char * aname)
 {
-  delete name;
+  delete [] name;
   if (aname)
     {
       name = new char[strlen(aname)+1];
diff --git a/Netgen/libsrc/general/flags.cpp b/Netgen/libsrc/general/flags.cpp
index 962c8152e8..9cce0cef2f 100644
--- a/Netgen/libsrc/general/flags.cpp
+++ b/Netgen/libsrc/general/flags.cpp
@@ -27,10 +27,10 @@ namespace netgen
   
   void Flags :: DeleteFlags ()
   {
-    int i;
-
-    for (i = 1; i <= strflags.Size(); i++)
-      delete strflags.Elem(i);
+    for (int i = 0; i < strflags.Size(); i++)
+      delete [] strflags[i];
+    for (int i = 0; i < numlistflags.Size(); i++)
+      delete numlistflags[i];
     strflags.DeleteAll();
     numflags.DeleteAll();
     defflags.DeleteAll();
diff --git a/Netgen/libsrc/general/hashtabl.cpp b/Netgen/libsrc/general/hashtabl.cpp
index 3081298e24..0485ab6045 100644
--- a/Netgen/libsrc/general/hashtabl.cpp
+++ b/Netgen/libsrc/general/hashtabl.cpp
@@ -8,6 +8,7 @@
    Abstract data type HASHTABLE
 */
 
+#include <algorithm>
 #include <mystdlib.h>
 #include <myadt.hpp>
 
diff --git a/Netgen/libsrc/general/hashtabl.hpp b/Netgen/libsrc/general/hashtabl.hpp
index 99d3257b84..b3e0e93f7e 100644
--- a/Netgen/libsrc/general/hashtabl.hpp
+++ b/Netgen/libsrc/general/hashtabl.hpp
@@ -48,7 +48,7 @@ public:
   ///
   inline const T & Get (const INDEX & ahash) const;
   ///
-  inline int Used (const INDEX & ahash) const;
+  inline bool Used (const INDEX & ahash) const;
   ///
   inline int GetNBags () const;
   ///
@@ -112,13 +112,21 @@ class INDEX_2_HASHTABLE : public BASE_INDEX_2_HASHTABLE
   
 public:
   ///
-  inline INDEX_2_HASHTABLE (int size);
+ INDEX_2_HASHTABLE (int size)
+   : BASE_INDEX_2_HASHTABLE (size), cont(size)
+  { ; }  
+
   ///
-  void SetSize(int s) {cont.SetSize(s); BaseSetSize(s);}
+  void SetSize(int s) 
+  { 
+    cont.SetSize(s); 
+    BaseSetSize(s);
+  }
+
   ///
-  inline void Set (const INDEX_2 & ahash, const T & acont)
-    {
-      int bnr = HashValue (ahash);
+  void Set (const INDEX_2 & ahash, const T & acont)
+  {
+    int bnr = HashValue (ahash);
       int pos = Position (bnr, ahash);
       if (pos)
 	cont.Set (bnr, pos, acont);
@@ -127,24 +135,140 @@ public:
 	  hash.Add1 (bnr, ahash);
 	  cont.Add1 (bnr, acont);
 	}    
-    }
+  }
+  
   ///
-  inline const T & Get (const INDEX_2 & ahash) const;
+  const T & Get (const INDEX_2 & ahash) const
+  {
+    int bnr = HashValue (ahash);
+    int pos = Position (bnr, ahash);
+    return cont.Get (bnr, pos);
+  }
+  
   ///
-  inline int Used (const INDEX_2 & ahash) const;
+  bool Used (const INDEX_2 & ahash) const
+  {
+    return (Position (HashValue (ahash), ahash)) ? 1 : 0;
+  }
+
   ///
-  inline int GetNBags () const;
+  int GetNBags () const
+  {
+    return cont.Size();
+  }
+  
   ///
-  inline int GetBagSize (int bnr) const;
+  int GetBagSize (int bnr) const
+  {
+    return cont.EntrySize (bnr);
+  }
+    
   ///
-  inline void GetData (int bnr, int colnr, 
-		       INDEX_2 & ahash, T & acont) const;
+  void GetData (int bnr, int colnr, 
+		INDEX_2 & ahash, T & acont) const
+  {
+    ahash = hash.Get(bnr, colnr);
+    acont = cont.Get(bnr, colnr);
+  }
+
   ///
-  inline void PrintMemInfo (ostream & ost) const;
+  void SetData (int bnr, int colnr, 
+		const INDEX_2 & ahash, const T & acont) 
+  {
+    hash.Set(bnr, colnr, ahash);
+    cont.Set(bnr, colnr, acont);
+  }
+  
+  ///
+  void PrintMemInfo (ostream & ost) const
+  {
+    ost << "Hash: " << endl;
+    hash.PrintMemInfo (ost);
+    ost << "Cont: " << endl;
+    cont.PrintMemInfo (ost);
+  }
+
+
+
+  class Iterator
+  {
+    const INDEX_2_HASHTABLE & ht;    
+    int bagnr, pos;
+  public:
+    Iterator (const INDEX_2_HASHTABLE & aht,
+	      int abagnr, int apos)
+      : ht(aht), bagnr(abagnr), pos(apos)
+    { ; }
+
+    int BagNr() const { return bagnr; }
+    int Pos() const { return pos; }
+
+    void operator++ (int)
+    {
+      // cout << "begin Operator ++: bagnr = " << bagnr << " -  pos = " << pos << endl;
+      pos++;
+      while (bagnr < ht.GetNBags() && 
+	     pos == ht.GetBagSize(bagnr+1))
+	{
+	  pos = 0;
+	  bagnr++;
+	}
+      // cout << "end Operator ++: bagnr = " << bagnr << " - pos = " << pos << endl;
+    }
+
+    bool operator != (int i) const
+    {
+      return bagnr != i;
+    }
+
+  };
+  
+  Iterator Begin () const
+  {
+    Iterator it(*this, 0, -1);
+    it++;
+    return it;
+  }
+
+  int End() const
+  {
+    return GetNBags();
+  }
+
+  void GetData (const Iterator & it,
+		INDEX_2 & ahash, T & acont) const
+  {
+    ahash = hash[it.BagNr()][it.Pos()];
+    acont = cont[it.BagNr()][it.Pos()];
+  }
+
+  const INDEX_2 & GetHash (const Iterator & it) const
+  { return hash[it.BagNr()][it.Pos()]; }
+
+  const T & GetData (const Iterator & it) const
+  { return cont[it.BagNr()][it.Pos()]; }
 };
 
 
 
+template <typename T> 
+inline ostream & operator<< (ostream & ost, const INDEX_2_HASHTABLE<T> & ht)
+{
+  for (typename INDEX_2_HASHTABLE<T>::Iterator it = ht.Begin();
+       it != ht.End(); it++)
+    {
+      ost << ht.GetHash(it) << ": " << ht.GetData(it) << endl;
+    }
+
+  return ost;
+}
+
+
+
+
+
+
+
 ///
 class BASE_INDEX_3_HASHTABLE
 {
@@ -197,7 +321,7 @@ public:
   ///
   inline const T & Get (const INDEX_3 & ahash) const;
   ///
-  inline int Used (const INDEX_3 & ahash) const;
+  inline bool Used (const INDEX_3 & ahash) const;
   ///
   inline int GetNBags () const;
   ///
@@ -311,7 +435,7 @@ public:
   ///
   inline const T & Get (const INDEX_2 & ahash) const;
   ///
-  inline int Used (const INDEX_2 & ahash) const;
+  inline bool Used (const INDEX_2 & ahash) const;
   ///
   inline void SetData (int pos, const INDEX_2 & ahash, const T & acont);
   ///
@@ -321,6 +445,8 @@ public:
   ///
   inline void GetData (int pos, T & acont) const;
   ///
+  const T & GetData (int pos) { return cont.Get(pos); }
+  ///
   inline void SetSize (int size);
   ///
   inline void PrintMemInfo (ostream & ost) const;
@@ -359,20 +485,30 @@ public:
   BASE_INDEX_3_CLOSED_HASHTABLE (int size);
 
   int Size() const { return hash.Size(); }
-  int UsedPos (int pos) const { return ! (hash.Get(pos).I1() == invalid); }
+  bool UsedPos (int pos) const { return ! (hash.Get(pos).I1() == invalid); }
   int UsedElements () const;
 
   ///
   int HashValue (const INDEX_3 & ind) const
-    {
-      return (ind.I1() + 15 * ind.I2() + 41 * ind.I3()) % hash.Size() + 1;
-    }
-
+  {
+    return (ind.I1() + 15 * ind.I2() + 41 * ind.I3()) % hash.Size() + 1;
+  }
+  
   int Position (const INDEX_3 & ind) const
   {
+    int i = HashValue(ind);
+    while (1)
+      {
+	if (hash.Get(i) == ind) return i;
+	if (hash.Get(i).I1() == invalid) return 0;
+	i++;
+	if (i > hash.Size()) i = 1;
+      }
+    /*
     int pos = HashValue (ind);
     if (hash.Get(pos) == ind) return pos;
     return Position2 (ind);
+    */
   }
 
   // returns 1, if new postion is created
@@ -417,7 +553,7 @@ public:
   ///
   inline const T & Get (const INDEX_3 & ahash) const;
   ///
-  inline int Used (const INDEX_3 & ahash) const;
+  inline bool Used (const INDEX_3 & ahash) const;
   ///
   inline void SetData (int pos, const INDEX_3 & ahash, const T & acont);
   ///
@@ -427,6 +563,8 @@ public:
   ///
   inline void GetData (int pos, T & acont) const;
   ///
+  inline const T & GetData (int pos) const;
+  ///
   inline void SetSize (int size);
   ///
   inline void PrintMemInfo (ostream & ost) const;
@@ -502,7 +640,7 @@ inline const T & INDEX_3_HASHTABLE<T> :: Get (const INDEX_3 & ahash) const
 }
 
 template<class T>
-inline int INDEX_3_HASHTABLE<T> :: Used (const INDEX_3 & ahash) const
+inline bool INDEX_3_HASHTABLE<T> :: Used (const INDEX_3 & ahash) const
 {
   return (Position (HashValue (ahash), ahash)) ? 1 : 0;
 }
@@ -609,7 +747,7 @@ inline const T & INDEX_HASHTABLE<T> :: Get (const INDEX & ahash) const
     }
 
 template<class T>
-inline int INDEX_HASHTABLE<T> :: Used (const INDEX & ahash) const
+inline bool INDEX_HASHTABLE<T> :: Used (const INDEX & ahash) const
     {
     return (Position (HashValue (ahash), ahash)) ? 1 : 0;
     }
@@ -648,82 +786,6 @@ inline void INDEX_HASHTABLE<T> :: PrintMemInfo (ostream & ost) const
     
     
     
-    
-    
-/****************** INDEX_2_HASHTABLE ****************************/    
-    
-    
-template<class T>
-inline INDEX_2_HASHTABLE<T> :: INDEX_2_HASHTABLE (int size)
-    : BASE_INDEX_2_HASHTABLE (size), cont(size)
-    {
-      ;
-    }
-
-/*
-template<class T>
-inline void INDEX_2_HASHTABLE<T> :: Set (const INDEX_2 & ahash, const T & acont)
-    {
-    int bnr = HashValue (ahash);
-    int pos = Position (bnr, ahash);
-    if (pos)
-      cont.Set (bnr, pos, acont);
-    else
-      {
-      hash.Add (bnr, ahash);
-      cont.Add (bnr, acont);
-      }
-    }
-*/
-
-template<class T>
-inline const T & INDEX_2_HASHTABLE<T> :: Get (const INDEX_2 & ahash) const
-    {
-    int bnr = HashValue (ahash);
-    int pos = Position (bnr, ahash);
-    return cont.Get (bnr, pos);
-    }
-
-template<class T>
-inline int INDEX_2_HASHTABLE<T> :: Used (const INDEX_2 & ahash) const
-    {
-    return (Position (HashValue (ahash), ahash)) ? 1 : 0;
-    }
-
-template<class T>
-inline int INDEX_2_HASHTABLE<T> :: GetNBags () const
-    {
-    return cont.Size();
-    }
-
-template<class T>
-inline int INDEX_2_HASHTABLE<T> :: GetBagSize (int bnr) const
-    {
-    return cont.EntrySize (bnr);
-    }
-
-template<class T>
-inline void INDEX_2_HASHTABLE<T> :: GetData (int bnr, int colnr, INDEX_2 & ahash, T & acont) const
-    {
-    ahash = hash.Get(bnr, colnr);
-    acont = cont.Get(bnr, colnr);
-    }
-
-
-
-template<class T>
-inline void INDEX_2_HASHTABLE<T> :: PrintMemInfo (ostream & ost) const
-  {
-    ost << "Hash: " << endl;
-    hash.PrintMemInfo (ost);
-    ost << "Cont: " << endl;
-    cont.PrintMemInfo (ost);
-  }
-
-
-
-
-
 
 
 
@@ -764,7 +826,7 @@ Get (const INDEX_2 & ahash) const
 }
 
 template<class T>
-inline int INDEX_2_CLOSED_HASHTABLE<T> :: 
+inline bool INDEX_2_CLOSED_HASHTABLE<T> :: 
 Used (const INDEX_2 & ahash) const
 {
   int pos = Position (ahash);
@@ -866,7 +928,7 @@ Get (const INDEX_3 & ahash) const
 }
 
 template<class T>
-inline int INDEX_3_CLOSED_HASHTABLE<T> :: 
+inline bool INDEX_3_CLOSED_HASHTABLE<T> :: 
 Used (const INDEX_3 & ahash) const
 {
   int pos = Position (ahash);
@@ -903,6 +965,13 @@ GetData (int pos, T & acont) const
   acont = cont.Get(pos);
 }
 
+template<class T>
+inline const T & INDEX_3_CLOSED_HASHTABLE<T> :: 
+GetData (int pos) const
+{
+  return cont.Get(pos);
+}
+
 
 template<class T>
 inline void INDEX_3_CLOSED_HASHTABLE<T> :: 
diff --git a/Netgen/libsrc/general/moveablemem.cpp b/Netgen/libsrc/general/moveablemem.cpp
index 0e6f894dea..cdfac0e7b8 100644
--- a/Netgen/libsrc/general/moveablemem.cpp
+++ b/Netgen/libsrc/general/moveablemem.cpp
@@ -36,7 +36,7 @@ namespace netgen
     pos = 0;
 
   ptr = 0;
-  name = 0;
+  name = NULL;
 
   if (s) Alloc(s);
 }
@@ -50,12 +50,20 @@ BaseMoveableMem :: ~BaseMoveableMem () throw()
   if (prev) prev->next = next;
   else first = next;
 
-  delete name;
+  if(name != NULL)
+    {
+      delete [] name;
+      name = NULL;
+    }
 }
 
 void BaseMoveableMem :: SetName (const char * aname)
 {
-  delete name;
+  if(name != NULL)
+    {
+      delete [] name;
+      name = NULL;
+    }
   if (aname)
     {
       name = new char[strlen(aname)+1];
@@ -192,7 +200,7 @@ void BaseMoveableMem :: Free () throw()
 void BaseMoveableMem :: Swap (BaseMoveableMem & m2) throw()
 {
   int hi;
-  BaseMoveableMem * hp;
+  // BaseMoveableMem * hp;
   char * cp;
   hi = size; size  = m2.size; m2.size = hi;
   hi = pos; pos = m2.pos; m2.pos = hi;
diff --git a/Netgen/libsrc/general/moveablemem.hpp b/Netgen/libsrc/general/moveablemem.hpp
index d156bb694c..92a67ad6da 100644
--- a/Netgen/libsrc/general/moveablemem.hpp
+++ b/Netgen/libsrc/general/moveablemem.hpp
@@ -90,8 +90,8 @@ public:
     BaseMoveableMem::Swap (m2);
   }
 protected:
-  MoveableMem (const MoveableMem & m);
-  MoveableMem & operator= (const MoveableMem & m);
+  MoveableMem (const MoveableMem & m) { ; }
+  MoveableMem & operator= (const MoveableMem & m) { ; }
 };
 
 #endif
diff --git a/Netgen/libsrc/general/mystring.cpp b/Netgen/libsrc/general/mystring.cpp
index b586202ba8..6b0f0cee74 100644
--- a/Netgen/libsrc/general/mystring.cpp
+++ b/Netgen/libsrc/general/mystring.cpp
@@ -41,14 +41,15 @@ void DefaultStringErrHandler()
 }
 
 void (*MyStr::ErrHandler)() = DefaultStringErrHandler;
-     
+
+  /*     
 MyStr::MyStr()
 {
   length = 0;
-  // str = new char[1];
   str = shortstr;
   str[0] = 0;
 }
+  */
 
 MyStr::MyStr(const char *s)
 {
@@ -61,6 +62,7 @@ MyStr::MyStr(const char *s)
   strcpy(str, s);
 }
 
+/*
 MyStr::MyStr(char s)
 {
   length = 1;
@@ -68,6 +70,7 @@ MyStr::MyStr(char s)
   str[0] = s;
   str[1] = (char)0;
 }
+*/
 
 MyStr::MyStr(const MyStr& s)
 {
diff --git a/Netgen/libsrc/general/mystring.hpp b/Netgen/libsrc/general/mystring.hpp
index 6517aad3fa..39851d1fad 100644
--- a/Netgen/libsrc/general/mystring.hpp
+++ b/Netgen/libsrc/general/mystring.hpp
@@ -93,6 +93,22 @@ private:
   static void(*ErrHandler)();
 };
 
+
+inline MyStr::MyStr()
+{
+  length = 0;
+  str = shortstr;
+  str[0] = 0;
+}
+
+inline MyStr::MyStr(char s)
+{
+  length = 1;
+  str = shortstr;
+  str[0] = s;
+  str[1] = (char)0;
+}
+
 inline MyStr::~MyStr()
 {
   if (length > SHORTLEN)
diff --git a/Netgen/libsrc/general/optmem.cpp b/Netgen/libsrc/general/optmem.cpp
index 3247e7e38d..c8f961b12a 100644
--- a/Netgen/libsrc/general/optmem.cpp
+++ b/Netgen/libsrc/general/optmem.cpp
@@ -32,10 +32,10 @@ namespace netgen
       delete [] bablocks[i];
   }
 
-  void * BlockAllocator :: Alloc ()
+  void BlockAllocator :: Alloc2 ()
   {
     //  return new char[size];
-    if (!freelist)
+    //    if (!freelist)
       {
 	// cout << "BlockAlloc: " << size*blocks << endl;
 	char * hcp = new char [size * blocks];
@@ -46,14 +46,18 @@ namespace netgen
 	*(void**)&(hcp[(blocks-1)*size]) = NULL;
 	freelist = hcp;
       }
+      /*
     void * p = freelist;
     freelist = *(void**)freelist;
     return p;
+      */
   }
 
+  /*
   void BlockAllocator :: Free (void * p)
   {
     *(void**)p = freelist;
     freelist = p;
   }
+  */
 }
diff --git a/Netgen/libsrc/general/optmem.hpp b/Netgen/libsrc/general/optmem.hpp
index a7a136c7c5..d9e5e2137e 100644
--- a/Netgen/libsrc/general/optmem.hpp
+++ b/Netgen/libsrc/general/optmem.hpp
@@ -26,9 +26,25 @@ public:
   ///
   ~BlockAllocator ();
   ///
-  void * Alloc ();
+  void * Alloc ()
+  {
+    if (!freelist)
+      Alloc2();
+
+    void * p = freelist;
+    freelist = *(void**)freelist;
+    return p;
+  }
   ///
-  void Free (void * p);
+  void Free (void * p)
+  {
+    *(void**)p = freelist;
+    freelist = p;
+  }
+  
+
+private:
+  void Alloc2 ();
 };
 
 
diff --git a/Netgen/libsrc/general/sort.cpp b/Netgen/libsrc/general/sort.cpp
index b4f2c8c0b4..264a132a74 100644
--- a/Netgen/libsrc/general/sort.cpp
+++ b/Netgen/libsrc/general/sort.cpp
@@ -9,6 +9,7 @@
 */
 
 
+#include <algorithm>
 #include <mystdlib.h>
 #include <myadt.hpp>
 
diff --git a/Netgen/libsrc/general/sort.hpp b/Netgen/libsrc/general/sort.hpp
index 63a85dad45..99c3291000 100644
--- a/Netgen/libsrc/general/sort.hpp
+++ b/Netgen/libsrc/general/sort.hpp
@@ -7,6 +7,7 @@
 /* Date:   07. Jan. 00                                                    */
 /**************************************************************************/
 
+
 // order(i) is sorted index of element i
 extern void Sort (const ARRAY<double> & values,
 		  ARRAY<int> & order);
diff --git a/Netgen/libsrc/general/symbolta.hpp b/Netgen/libsrc/general/symbolta.hpp
index e429c7d19b..6b8916e8ea 100644
--- a/Netgen/libsrc/general/symbolta.hpp
+++ b/Netgen/libsrc/general/symbolta.hpp
@@ -9,9 +9,9 @@
 /**************************************************************************/
 
 /**
-  Base class for the generic SYMBOLTABLE.
-  An array of identifiers is maintained.
- */
+   Base class for the generic SYMBOLTABLE.
+   An array of identifiers is maintained.
+*/
 class BASE_SYMBOLTABLE
 {
 protected:
@@ -31,14 +31,15 @@ public:
 
 
 /** 
-   Abstract data type Symbol Table.
+    Abstract data type Symbol Table.
    
-   To a string an value of the generic type T is associated.
-   The string is not copied into the symbol table class!
+    To a string an value of the generic type T is associated.
+    The string is not copied into the symbol table class!
 */
 template <class T>
 class SYMBOLTABLE : public BASE_SYMBOLTABLE
 {
+private:
   /// Associated data
   ARRAY <T> data;
   
@@ -51,7 +52,7 @@ public:
   inline T & Elem (const char * name);
   /// Returns reference to i-th element
   inline T & Elem (int i) 
-    { return data.Elem(i); }
+  { return data.Elem(i); }
   /// Returns element, error if not used
   inline const T & Get (const char * name) const;
   /// Returns i-th element
@@ -66,9 +67,9 @@ public:
   inline void DeleteAll ();
 
   inline T & operator[] (int i) 
-    { return data[i]; }
+  { return data[i]; }
   inline const T & operator[] (int i) const
-    { return data[i]; }
+  { return data[i]; }
 
 private:
   /// Prevents from copying symboltable by pointer assignment
@@ -80,79 +81,78 @@ private:
 
 template <class T>
 inline SYMBOLTABLE<T> :: SYMBOLTABLE () 
-  { 
-    ;
-  }
+{ 
+  ;
+}
 
 
 template <class T>
 inline INDEX SYMBOLTABLE<T> :: Size() const
-  {
+{
   return data.Size();
-  }
+}
 
 template <class T>
 inline T & SYMBOLTABLE<T> :: Elem (const char * name)
-  {
-  int i;
-  i = Index (name);
+{
+  int i = Index (name);
   if (i) 
     return data.Elem (i);
   else 
     return data.Elem(1);
-  }
+}
 
 template <class T>
 inline const T & SYMBOLTABLE<T> :: Get (const char * name) const
-  {
+{
   int i;
   i = Index (name);
   if (i) 
     return data.Get(i);
   else 
     return data.Get(1);
-  }
+}
 
 template <class T>
 inline const T & SYMBOLTABLE<T> :: Get (int i) const
-  {
+{
   return data.Get(i);
-  }
+}
 
 template <class T>
 inline const char* SYMBOLTABLE<T> :: GetName (int i) const
-  {
+{
   return names.Get(i);
-  }
+}
 
 template <class T>
 inline void SYMBOLTABLE<T> :: Set (const char * name, const T & el)
-  {
+{
   int i;
   i = Index (name);
   if (i) 
     data.Set(i, el);
   else
     {
-    data.Append (el);
-    char * hname = new char [strlen (name) + 1];
-    strcpy (hname, name);
-    names.Append (hname);
+      data.Append (el);
+      char * hname = new char [strlen (name) + 1];
+      strcpy (hname, name);
+      names.Append (hname);
     }
-  }
+}
 
 template <class T>
 inline int SYMBOLTABLE<T> :: Used (const char * name) const
-  {
+{
   return (Index(name)) ? 1 : 0;
-  }
+}
 
 template <class T>
 inline void SYMBOLTABLE<T> :: DeleteAll () 
-  {	
+{	
   DelNames();
   data.DeleteAll();
-  }
+}
 
 
 #endif
diff --git a/Netgen/libsrc/general/table.cpp b/Netgen/libsrc/general/table.cpp
index 7dcd5557d3..ed84789bf9 100644
--- a/Netgen/libsrc/general/table.cpp
+++ b/Netgen/libsrc/general/table.cpp
@@ -62,12 +62,11 @@ namespace netgen
   
   void BASE_TABLE :: SetSize (int size)
   {
-    int i;
-    for (i = 0; i < data.Size(); i++)
+    for (int i = 0; i < data.Size(); i++)
       delete [] (char*)data[i].col;
     
     data.SetSize(size);
-    for (i = 0; i < size; i++)
+    for (int i = 0; i < size; i++)
       {
 	data[i].maxsize = 0;
 	data[i].size = 0;
@@ -77,37 +76,22 @@ namespace netgen
 
   void BASE_TABLE :: IncSize2 (int i, int elsize)
   {
-    if (i < 1 || i > data.Size())
+#ifdef DEBUG
+    if (i < 0 || i >= data.Size())
       {
 	MyError ("BASE_TABLE::Inc: Out of range");
 	return;
       }
-  
-    linestruct & line = data.Elem (i);
-  
+#endif
+    
+    linestruct & line = data[i];
     if (line.size == line.maxsize)
       {
-	/*
-	  static int totalloc = 0, cnt = 0;
-	  totalloc += (line.maxsize+5) * elsize;
-	  cnt ++;
-
-	  if (cnt % 100000 == 0)
-	  cout << "base_table: total alloc = " << totalloc << endl;
-	*/
-
 	void * p = new char [(line.maxsize+5) * elsize];
-	// mem_total_alloc_table += (line.maxsize+5) * elsize;
-
-
-	if (!p)
-	  {
-	    MyError ("BASE_TABLE::Inc: Out of memory");
-	    return;
-	  }
       
 	memcpy (p, line.col, line.maxsize * elsize);
 	delete [] (char*)line.col;
+
 	line.col = p;
 	line.maxsize += 5;
       }
@@ -115,64 +99,55 @@ namespace netgen
     line.size++;
   }
 
+
+
+  /*
   void BASE_TABLE :: DecSize (int i)
   {
-    if (i < 1 || i > data.Size())
+#ifdef DEBUG
+    if (i < 0 || i >= data.Size())
       {
 	MyError ("BASE_TABLE::Dec: Out of range");
 	return;
       }
+#endif
+
+    linestruct & line = data[i];
   
-    linestruct & line = data.Elem (i);
-  
+#ifdef DEBUG
     if (line.size == 0)
       {
 	MyError ("BASE_TABLE::Dec: EntrySize < 0");
 	return;      
       }
+#endif
   
     line.size--;
   }
-
-
-  /*
-  void BASE_TABLE :: IncSizePrepare (int i)
-  {
-    data.Elem(i).maxsize++;
-  }
   */
 
 
+
   void BASE_TABLE :: AllocateElementsOneBlock (int elemsize)
   {
-    int i, cnt = 0;
+    int cnt = 0;
     int n = data.Size();
 
-    for (i = 1; i <= n; i++)
-      cnt += data.Get(i).maxsize;
+    for (int i = 0; i < n; i++)
+      cnt += data[i].maxsize;
     oneblock = new char[elemsize * cnt];
-    // mem_total_alloc_table += elemsize * cnt;
-
-    //  cout << "Allocate oneblock, mem = " << (elemsize * cnt) << endl;
 
     cnt = 0;
-    for (i = 1; i <= n; i++)
+    for (int i = 0; i < n; i++)
       {
-	data.Elem(i).size = 0;
-
-	data.Elem(i).col = &oneblock[elemsize * cnt];
-	cnt += data.Elem(i).maxsize;
+	data[i].size = 0;
+	data[i].col = &oneblock[elemsize * cnt];
+	cnt += data[i].maxsize;
       }
-  
   }
 
 
 
-
-
-
-
-
   int BASE_TABLE :: AllocatedElements () const
   {
     int els = 0;
@@ -180,7 +155,7 @@ namespace netgen
       els += data[i].maxsize;
     return els;
   }
-
+  
   int BASE_TABLE :: UsedElements () const
   {
     int els = 0;
diff --git a/Netgen/libsrc/general/table.hpp b/Netgen/libsrc/general/table.hpp
index 21c6d41a60..b986049b00 100644
--- a/Netgen/libsrc/general/table.hpp
+++ b/Netgen/libsrc/general/table.hpp
@@ -37,18 +37,19 @@ public:
   ~BASE_TABLE ();
   ///
   void SetSize (int size);
-  ///
+
+  /// increment size of entry i by one, i is 0-based
   void IncSize (int i, int elsize)
   {
-    if (data.Elem(i).size < data.Elem(i).maxsize)
-      data.Elem(i).size++;
+    if (data[i].size < data[i].maxsize)
+      data[i].size++;
     else
       IncSize2 (i, elsize);
   }
   ///
   void IncSize2 (int i, int elsize);
 
-  void DecSize (int i);
+  //  void DecSize (int i);
 
   ///
   void AllocateElementsOneBlock (int elemsize);
@@ -92,29 +93,37 @@ public:
   }
 
 
-  /// Inserts element acont into row i, 1-based. Does not test if already used.
+  /// Inserts element acont into row i, BASE-based. Does not test if already used.
   inline void Add (int i, const T & acont)
   {
-    IncSize (i-BASE+1, sizeof (T));
+    IncSize (i-BASE, sizeof (T));
     ((T*)data[i-BASE].col)[data[i-BASE].size-1] = acont;
   }
 
-  ///
-  void IncSizePrepare (int i)
-  {
-    data[i-BASE].maxsize++;
-  }
-
 
   /// Inserts element acont into row i, 1-based. Does not test if already used.
   inline void Add1 (int i, const T & acont)
   {
-    IncSize (i, sizeof (T));
+    IncSize (i-1, sizeof (T));
     ((T*)data.Elem(i).col)[data.Elem(i).size-1] = acont;
   }
   
-  /// Inserts element acont into row i. Does not test if already used, assumes to have mem
+  ///
+  void IncSizePrepare (int i)
+  {
+    data[i-BASE].maxsize++;
+  }
+
+
+  /// Inserts element acont into row i. BASE-based. Does not test if already used, assumes to have mem
   inline void AddSave (int i, const T & acont)
+    {
+      ((T*)data[i-BASE].col)[data[i-BASE].size] = acont;
+      data[i-BASE].size++;
+    }
+
+  /// Inserts element acont into row i. 1-based. Does not test if already used, assumes to have mem
+  inline void AddSave1 (int i, const T & acont)
     {
       ((T*)data.Elem(i).col)[data.Elem(i).size] = acont;
       data.Elem(i).size++;
@@ -123,7 +132,7 @@ public:
   /// Inserts element acont into row i. Does not test if already used.
   inline void AddEmpty (int i)
   {
-    IncSize (i, sizeof (T));
+    IncSize (i-BASE, sizeof (T));
   }
 
   /** Set the nr-th element in the i-th row to acont.
@@ -153,9 +162,10 @@ public:
   inline int EntrySize (int i) const
     { return data.Get(i).size; }
 
+  /*
   inline void DecEntrySize (int i)
     { DecSize(i); }
-
+  */
   void AllocateElementsOneBlock ()
     { BASE_TABLE::AllocateElementsOneBlock (sizeof(T)); }
 
diff --git a/Netgen/libsrc/general/template.hpp b/Netgen/libsrc/general/template.hpp
index 47c7a1ad2e..4bf983c06b 100644
--- a/Netgen/libsrc/general/template.hpp
+++ b/Netgen/libsrc/general/template.hpp
@@ -142,15 +142,28 @@ public:
     { return i[0] == in2.i[0] && i[1] == in2.i[1]; }
 
   ///
-  void Sort ()
-    {
-      if (i[0] > i[1]) 
-	{
-	  INDEX hi = i[0];
-	  i[0] = i[1];
-	  i[1] = hi;
-	}
-    }
+
+
+  INDEX_2 Sort ()
+  {
+    if (i[0] > i[1]) 
+      {
+	INDEX hi = i[0];
+	i[0] = i[1];
+	i[1] = hi;
+      }
+    return *this;
+  }
+
+  static INDEX_2 Sort (int i1, int i2)
+  {
+    if (i1 > i2)
+      return INDEX_2 (i2,i1);
+    else
+      return INDEX_2 (i1,i2);
+  }
+
+
   ///
   INDEX & I1 () { return i[0]; }
   ///
@@ -189,12 +202,25 @@ public:
   INDEX_3 (const INDEX_3 & in2)
     { i[0] = in2.i[0]; i[1] = in2.i[1]; i[2] = in2.i[2]; }
 
+
+  static INDEX_3 Sort (INDEX_3 i3)
+  {
+    return i3.Sort();
+  }
+
+  static INDEX_3 Sort (int i1, int i2, int i3)
+  {
+    INDEX_3 i(i1, i2, i3);
+    return i.Sort();
+  }
+
   ///
-  void Sort ()
+  INDEX_3 Sort ()
   {
     if (i[0] > i[1]) Swap (i[0], i[1]);
     if (i[1] > i[2]) Swap (i[1], i[2]);
     if (i[0] > i[1]) Swap (i[0], i[1]);
+    return *this;
   }
 
   ///
diff --git a/Netgen/libsrc/geom2d/genmesh2d.cpp b/Netgen/libsrc/geom2d/genmesh2d.cpp
index db31208940..34f93b66ee 100644
--- a/Netgen/libsrc/geom2d/genmesh2d.cpp
+++ b/Netgen/libsrc/geom2d/genmesh2d.cpp
@@ -22,6 +22,8 @@ namespace netgen
     int i, j, domnr;
     double elto0, minx, miny, maxx, maxy;
 
+    //    mp.Print(*testout);
+
     PointIndex pi;
     SegmentIndex si;
     SurfaceElementIndex sei;
@@ -43,13 +45,30 @@ namespace netgen
 
     geometry.PartitionBoundary (h, *mesh);
 
+    for (i = 0; i < geometry.GetNP(); i++)
+      if (geometry.GetPoint(i).hpref)
+	{
+	  double mindist = 1e99;
+	  PointIndex mpi;
+	  Point<2> gp = geometry.GetPoint(i);
+	  Point<3> gp3(gp(0), gp(1), 0);
+	  for (PointIndex pi = PointIndex::BASE; 
+	       pi < mesh->GetNP()+PointIndex::BASE; pi++)
+	    if (Dist2(gp3, (*mesh)[pi]) < mindist)
+	      {
+		mpi = pi;
+		mindist = Dist2(gp3, (*mesh)[pi]);
+	      }
+	  (*mesh)[mpi].SetSingular();
+	}
+
+
     int maxdomnr = 0;
     for (si = 0; si < mesh->GetNSeg(); si++)
       {
 	if ( (*mesh)[si].domin > maxdomnr) maxdomnr = (*mesh)[si].domin;
 	if ( (*mesh)[si].domout > maxdomnr) maxdomnr = (*mesh)[si].domout;
       }
-  
 
     mesh->ClearFaceDescriptors();
     for (i = 1; i <= maxdomnr; i++)
@@ -111,10 +130,15 @@ namespace netgen
 
     mp.optsteps2d = hsteps;
 
-
-
     mesh->Compress();
     mesh -> SetNextMajorTimeStamp();
+
+
+#ifdef OPENGL
+    extern void Render();
+    Render();
+#endif
+
   }
 
 
diff --git a/Netgen/libsrc/geom2d/geom2dmesh.cpp b/Netgen/libsrc/geom2d/geom2dmesh.cpp
index 3ca3275d29..09664da239 100644
--- a/Netgen/libsrc/geom2d/geom2dmesh.cpp
+++ b/Netgen/libsrc/geom2d/geom2dmesh.cpp
@@ -7,90 +7,49 @@
 namespace netgen
 {
 
-Refinement2d :: Refinement2d (const SplineGeometry2d & ageometry)
-  : Refinement(), geometry(ageometry)
-{
-  ;
-}
-
-Refinement2d :: ~Refinement2d ()
-{
-  ;
-}
+  Refinement2d :: Refinement2d (const SplineGeometry2d & ageometry)
+    : Refinement(), geometry(ageometry)
+  {
+    ;
+  }
+
+  Refinement2d :: ~Refinement2d ()
+  {
+    ;
+  }
   
 
+  void Refinement2d :: 
+  PointBetween (const Point3d & p1, const Point3d & p2, double secpoint,
+		int surfi, 
+		const PointGeomInfo & gi1, 
+		const PointGeomInfo & gi2,
+		Point3d & newp, PointGeomInfo & newgi)
+  {
+    newp = p1+secpoint*(p2-p1);
+    newgi.trignum = 1;
+  }
+
+
+
+  void Refinement2d :: 
+  PointBetween (const Point3d & p1, const Point3d & p2, double secpoint, 
+		int surfi1, int surfi2, 
+		const EdgePointGeomInfo & ap1, 
+		const EdgePointGeomInfo & ap2,
+		Point3d & newp, EdgePointGeomInfo & newgi)
+  {
+    Point<2> p2d;
+  
+    p2d = geometry.GetSplines().Get(ap1.edgenr) -> 
+      GetPoint (((1-secpoint)*ap1.dist+secpoint*ap2.dist));
+  
+    //  (*testout) << "refine 2d line, ap1.dist, ap2.dist = " << ap1.dist << ", " << ap2.dist << endl;
+    //  (*testout) << "p1, p2 = " << p1 << p2 << ", newp = " << p2d << endl;
 
-void Refinement2d :: 
-PointBetween (const Point3d & p1, const Point3d & p2, double secpoint,
-	      int surfi, 
-	      const PointGeomInfo & gi1, 
-	      const PointGeomInfo & gi2,
-	      Point3d & newp, PointGeomInfo & newgi)
-{
-  newp = p1+secpoint*(p2-p1);
-  newgi.trignum = 1;
-  /*
-  if (surfi)
-    {
-      //      (*testout) << "surfi1 = " << surfi << endl;
-      //      (*testout) << "np = " << newp << endl;
-      geometry.GetSurface (surfi) -> Project (newp);
-      newgi.trignum = 1;
-      //      (*testout) << "np2 = " << newp << endl;
-    }
-  */
-}
-
-
-
-void Refinement2d :: 
-PointBetween (const Point3d & p1, const Point3d & p2, double secpoint, 
-	      int surfi1, int surfi2, 
-	      const EdgePointGeomInfo & ap1, 
-	      const EdgePointGeomInfo & ap2,
-	      Point3d & newp, EdgePointGeomInfo & newgi)
-{
-  Point<2> p2d;
-
-  p2d = geometry.GetSplines().Get(ap1.edgenr) -> 
-    GetPoint (((1-secpoint)*ap1.dist+secpoint*ap2.dist));
-
-
-/*
-
-  // test routine: general function on first edge
-
-  double x = secpoint;
-  if (ap1.edgenr == 1)
-      p2d = Point<2>((1-secpoint),fabs(x-0.5) < 0.2 ? (1+cos((x-0.5)*M_PI/0.2))*0.1 : 0);
-
-*/
-
-  //  (*testout) << "refine 2d line, ap1.dist, ap2.dist = " << ap1.dist << ", " << ap2.dist << endl;
-  //  (*testout) << "p1, p2 = " << p1 << p2 << ", newp = " << p2d << endl;
-
-  //  newp = Center (p1, p2);
-  newp = Point3d (p2d(0), p2d(1), 0);
-  newgi.edgenr = ap1.edgenr;
-  newgi.dist = ((1-secpoint)*ap1.dist+secpoint*ap2.dist);
-
-  /*
-  if (surfi1 && surfi2 && surfi1 != surfi2)
-    {
-      //      (*testout) << "surfi1 = " << surfi1 << " surfi2 = " << surfi2 << endl;
-      //      (*testout) << "np = " << newp << endl;
-      //  geometry.GetSurface (surfi1) -> Project (newp);
-      ProjectToEdge (geometry.GetSurface(surfi1), 
-		     geometry.GetSurface(surfi2), 
-		     newp);
-      newgi.edgenr = 1;
-      //      (*testout) << "np2 = " << newp << endl;
-    }
-  else if (surfi1)
-    {
-      geometry.GetSurface (surfi1) -> Project (newp);
-    }
-  */
-};
+    newp = Point3d (p2d(0), p2d(1), 0);
+    newgi.edgenr = ap1.edgenr;
+    newgi.dist = ((1-secpoint)*ap1.dist+secpoint*ap2.dist);
+  };
 
 }
diff --git a/Netgen/libsrc/geom2d/spline2d.cpp b/Netgen/libsrc/geom2d/spline2d.cpp
index 18c0362f41..8609e064d1 100644
--- a/Netgen/libsrc/geom2d/spline2d.cpp
+++ b/Netgen/libsrc/geom2d/spline2d.cpp
@@ -14,337 +14,382 @@ namespace netgen
 #include "spline2d.hpp"
 
 
-void CalcPartition (double l, double h, double r1, double r2,
-		    double ra, double elto0, ARRAY<double> & points);
-
-
-
-// calculates length of spline-curve
-double SplineSegment :: Length () const
-{
-  Point<2> p, pold;
-
-  int i, n = 100;
-  double dt = 1.0 / n;
-
-  pold = GetPoint (0);
-
-  double l = 0;
-  for (i = 1; i <= n; i++)
-    {
-      p = GetPoint (i * dt);
-      l += Dist (p, pold);
-      pold = p;
-    }
-  return l;
-}
-
-
-
-// partitionizes spline curve
-void SplineSegment :: Partition (double h, double elto0,
-				 Mesh & mesh, int segnr) const
-{
-  int i, j;
-  double l, r1, r2, ra;
-  double lold, dt, frac;
-  int n = 100;
-  Point<2> p, pold, mark, oldmark;
-  ARRAY<double> curvepoints;
-  double edgelength, edgelengthold;
-  l = Length();
-
-  r1 = StartPI().refatpoint;
-  r2 = EndPI().refatpoint;
-  ra = reffak;
-
-  //  cout << "Partition, l = " << l << ", h = " << h << endl;
-  CalcPartition (l, h, r1, r2, ra, elto0, curvepoints);
-  //  cout << "curvepoints = " << curvepoints << endl;
-
-  dt = 1.0 / n;
-
-  l = 0;
-  j = 1;
-
-  pold = GetPoint (0);
-  lold = 0;
-  oldmark = pold;
-  edgelengthold = 0;
-
-  for (i = 1; i <= n; i++)
-    {
-      p = GetPoint (i*dt);
-      l = lold + Dist (p, pold);
-      while (j < curvepoints.Size() && (l >= curvepoints[j] || i == n))
-	{
-	  frac = (curvepoints[j]-lold) / (l-lold);
-	  mark = pold + frac * (p-pold);
-	  edgelength = i*dt + (frac-1)*dt;
+  void CalcPartition (double l, double h, double r1, double r2,
+		      double ra, double elto0, ARRAY<double> & points);
+
+
+
+  // calculates length of spline-curve
+  double SplineSegment :: Length () const
+  {
+    Point<2> p, pold;
+
+    int i, n = 100;
+    double dt = 1.0 / n;
+
+    pold = GetPoint (0);
+
+    double l = 0;
+    for (i = 1; i <= n; i++)
+      {
+	p = GetPoint (i * dt);
+	l += Dist (p, pold);
+	pold = p;
+      }
+    return l;
+  }
+
+
+
+  // partitionizes spline curve
+  void SplineSegment :: Partition (double h, double elto0,
+				   Mesh & mesh, Point3dTree & searchtree, int segnr) const
+  {
+    int i, j;
+    double l, r1, r2, ra;
+    double lold, dt, frac;
+    int n = 100;
+    Point<2> p, pold, mark, oldmark;
+    ARRAY<double> curvepoints;
+    double edgelength, edgelengthold;
+    l = Length();
+
+    r1 = StartPI().refatpoint;
+    r2 = EndPI().refatpoint;
+    ra = reffak;
+
+    //  cout << "Partition, l = " << l << ", h = " << h << endl;
+    CalcPartition (l, h, r1, r2, ra, elto0, curvepoints);
+    //  cout << "curvepoints = " << curvepoints << endl;
+
+    dt = 1.0 / n;
+
+    l = 0;
+    j = 1;
+
+    pold = GetPoint (0);
+    lold = 0;
+    oldmark = pold;
+    edgelengthold = 0;
+    ARRAY<int> locsearch;
+
+    for (i = 1; i <= n; i++)
+      {
+	p = GetPoint (i*dt);
+	l = lold + Dist (p, pold);
+	while (j < curvepoints.Size() && (l >= curvepoints[j] || i == n))
 	  {
-	    PointIndex pi1 = -1, pi2 = -1;
+	    frac = (curvepoints[j]-lold) / (l-lold);
+	    mark = pold + frac * (p-pold);
+	    edgelength = i*dt + (frac-1)*dt;
+	    {
+	      PointIndex pi1 = -1, pi2 = -1;
 	  
-	    Point3d mark3(mark(0), mark(1), 0);
-	    Point3d oldmark3(oldmark(0), oldmark(1), 0);
+	      Point3d mark3(mark(0), mark(1), 0);
+	      Point3d oldmark3(oldmark(0), oldmark(1), 0);
+
+	      Vec<3> v (1e-4*h, 1e-4*h, 1e-4*h);
+	      searchtree.GetIntersecting (oldmark3 - v, oldmark3 + v, locsearch);
+	      if (locsearch.Size()) pi1 = locsearch[0];
+	      
+	      searchtree.GetIntersecting (mark3 - v, mark3 + v, locsearch);
+	      if (locsearch.Size()) pi2 = locsearch[0];
+	      /*	    
+			   for (PointIndex pk = PointIndex::BASE; 
+			   pk < mesh.GetNP()+PointIndex::BASE; pk++)
+			   {
+			   if (Dist (mesh[pk], oldmark3) < 1e-4 * h) pi1 = pk;
+			   if (Dist (mesh[pk], mark3) < 1e-4 * h) pi2 = pk;
+			   }
+	      */
 	    
-	    for (PointIndex pk = PointIndex::BASE; 
-		 pk < mesh.GetNP()+PointIndex::BASE; pk++)
-	      {
-		if (Dist (mesh[pk], oldmark3) < 1e-4 * h) pi1 = pk;
-		if (Dist (mesh[pk], mark3) < 1e-4 * h) pi2 = pk;
-	      }
-
-	    //	    cout << "pi1 = " << pi1 << endl;
-	    //	    cout << "pi2 = " << pi2 << endl;
-	    
-	    if (pi1 == -1) pi1 = mesh.AddPoint(oldmark3);
-	    if (pi2 == -1) pi2 = mesh.AddPoint(mark3);
 
-	    // cout << "pi1 = " << pi1 << endl;
-	    // cout << "pi2 = " << pi2 << endl;
+	      //	    cout << "pi1 = " << pi1 << endl;
+	      //	    cout << "pi2 = " << pi2 << endl;
+	    
+	      if (pi1 == -1)
+		{
+		  pi1 = mesh.AddPoint(oldmark3);
+		  searchtree.Insert (oldmark3, pi1);
+		}
+	      if (pi2 == -1)
+		{
+		  pi2 = mesh.AddPoint(mark3);
+		  searchtree.Insert (mark3, pi2);
+		}
+
+	      // cout << "pi1 = " << pi1 << endl;
+	      // cout << "pi2 = " << pi2 << endl;
 	  
-	    Segment seg;
-	    seg.edgenr = segnr;
-	    seg.si = bc; // segnr;
-	    seg.p1 = pi1;
-	    seg.p2 = pi2;
-	    seg.domin = leftdom;
-	    seg.domout = rightdom;
-	    seg.epgeominfo[0].edgenr = segnr;
-	    seg.epgeominfo[0].dist = edgelengthold;
-	    seg.epgeominfo[1].edgenr = segnr;
-	    seg.epgeominfo[1].dist = edgelength;
-
-	    mesh.AddSegment (seg);
-	  }
+	      Segment seg;
+	      seg.edgenr = segnr;
+	      seg.si = bc; // segnr;
+	      seg.p1 = pi1;
+	      seg.p2 = pi2;
+	      seg.domin = leftdom;
+	      seg.domout = rightdom;
+	      seg.epgeominfo[0].edgenr = segnr;
+	      seg.epgeominfo[0].dist = edgelengthold;
+	      seg.epgeominfo[1].edgenr = segnr;
+	      seg.epgeominfo[1].dist = edgelength;
+	      seg.singedge_left = hpref_left;
+	      seg.singedge_right = hpref_right;
+	      mesh.AddSegment (seg);
+	    }
 	
-	  oldmark = mark;
-	  edgelengthold = edgelength;
-	  j++;
-	}
+	    oldmark = mark;
+	    edgelengthold = edgelength;
+	    j++;
+	  }
     
-      pold = p;
-      lold = l;
-    }
-}
+	pold = p;
+	lold = l;
+      }
+  }
 
 
-void SplineSegment :: GetPoints (int n, ARRAY<Point<2> > & points)
-{
-  points.SetSize (n);
-  if (n >= 2)
-    for (int i = 0; i < n; i++)
-      points[i] = GetPoint(double(i) / (n-1));
-}
+  void SplineSegment :: GetPoints (int n, ARRAY<Point<2> > & points)
+  {
+    points.SetSize (n);
+    if (n >= 2)
+      for (int i = 0; i < n; i++)
+	points[i] = GetPoint(double(i) / (n-1));
+  }
 
 
 
-/* 
-   Implementation of line-segment from p1 to p2
-*/
+  /* 
+     Implementation of line-segment from p1 to p2
+  */
 
 
-LineSegment :: LineSegment (const GeomPoint2d & ap1, 
-			    const GeomPoint2d & ap2)
-  : p1(ap1), p2(ap2)
-{
-  ;
-}
+  LineSegment :: LineSegment (const GeomPoint2d & ap1, 
+			      const GeomPoint2d & ap2)
+    : p1(ap1), p2(ap2)
+  {
+    ;
+  }
 
 
-Point<2> LineSegment :: GetPoint (double t) const
-{
-  return p1 + t * (p2 - p1);
-}
+  Point<2> LineSegment :: GetPoint (double t) const
+  {
+    return p1 + t * (p2 - p1);
+  }
 
-double LineSegment :: Length () const
-{
-  return Dist (p1, p2);
-}
+  double LineSegment :: Length () const
+  {
+    return Dist (p1, p2);
+  }
 
 
-void LineSegment :: PrintCoeff (ostream & ost) const
-{
-  double dx = p2(0) - p1(0);
-  double dy = p2(1) - p1(1);
-  ost << "0 0 0 " <<  dy << " " << -dx << " " 
-      << dx * p1(1) - dy * p1(0) << endl;
-}
+  void LineSegment :: PrintCoeff (ostream & ost) const
+  {
+    double dx = p2(0) - p1(0);
+    double dy = p2(1) - p1(1);
+    ost << "0 0 0 " <<  dy << " " << -dx << " " 
+	<< dx * p1(1) - dy * p1(0) << endl;
+  }
 
 
 
 
 
-SplineSegment3 :: SplineSegment3 (const GeomPoint2d & ap1, 
-				  const GeomPoint2d & ap2,
-				  const GeomPoint2d & ap3)
-  : p1(ap1), p2(ap2), p3(ap3)
-{
-  ;
-}
+  SplineSegment3 :: SplineSegment3 (const GeomPoint2d & ap1, 
+				    const GeomPoint2d & ap2,
+				    const GeomPoint2d & ap3)
+    : p1(ap1), p2(ap2), p3(ap3)
+  {
+    ;
+  }
 
-Point<2> SplineSegment3 :: GetPoint (double t) const
-{
-  double x, y, w;
-  double b1, b2, b3;
+  Point<2> SplineSegment3 :: GetPoint (double t) const
+  {
+    double x, y, w;
+    double b1, b2, b3;
 
-  b1 = (1-t)*(1-t);
-  b2 = sqrt(2.0) * t * (1-t);
-  b3 = t * t;
+    b1 = (1-t)*(1-t);
+    b2 = sqrt(2.0) * t * (1-t);
+    b3 = t * t;
 
-  x = p1(0) * b1 + p2(0) * b2 + p3(0) * b3;
-  y = p1(1) * b1 + p2(1) * b2 + p3(1) * b3;
-  w = b1 + b2 + b3;
+    x = p1(0) * b1 + p2(0) * b2 + p3(0) * b3;
+    y = p1(1) * b1 + p2(1) * b2 + p3(1) * b3;
+    w = b1 + b2 + b3;
 
-  return Point<2> (x/w, y/w);
-}
+    return Point<2> (x/w, y/w);
+  }
 
 
-void SplineSegment3 :: PrintCoeff (ostream & ost) const
-{
-  double t;
-  int i;
-  Point<2> p;
-  DenseMatrix a(6, 6);
-  DenseMatrix ata(6, 6);
-  Vector u(6), f(6);
-
-  //  ata.SetSymmetric(1);
-
-  t = 0;
-  for (i = 1; i <= 5; i++, t += 0.25)
-    {
-      p = GetPoint (t);
-      a.Elem(i, 1) = p(0) * p(0);
-      a.Elem(i, 2) = p(1) * p(1);
-      a.Elem(i, 3) = p(0) * p(1);
-      a.Elem(i, 4) = p(0);
-      a.Elem(i, 5) = p(1);
-      a.Elem(i, 6) = 1;
-    }
-  a.Elem(6, 1) = 1;
-
-  CalcAtA (a, ata);
-
-  u = 0;
-  u.Elem(6) = 1;
-  a.MultTrans (u, f);
-  ata.Solve (f, u);
+  void SplineSegment3 :: PrintCoeff (ostream & ost) const
+  {
+    double t;
+    int i;
+    Point<2> p;
+    DenseMatrix a(6, 6);
+    DenseMatrix ata(6, 6);
+    Vector u(6), f(6);
+
+    //  ata.SetSymmetric(1);
+
+    t = 0;
+    for (i = 1; i <= 5; i++, t += 0.25)
+      {
+	p = GetPoint (t);
+	a.Elem(i, 1) = p(0) * p(0);
+	a.Elem(i, 2) = p(1) * p(1);
+	a.Elem(i, 3) = p(0) * p(1);
+	a.Elem(i, 4) = p(0);
+	a.Elem(i, 5) = p(1);
+	a.Elem(i, 6) = 1;
+      }
+    a.Elem(6, 1) = 1;
+
+    CalcAtA (a, ata);
+
+    u = 0;
+    u.Elem(6) = 1;
+    a.MultTrans (u, f);
+    ata.Solve (f, u);
   
-  for (i = 1; i <= 6; i++)
-    ost << u.Get(i) << "  ";
-  ost << endl;
-}
+    for (i = 1; i <= 6; i++)
+      ost << u.Get(i) << "  ";
+    ost << endl;
+  }
 
 
 
 
-//########################################################################
-//		circlesegment
+  //########################################################################
+  //		circlesegment
 
-CircleSegment :: CircleSegment (const GeomPoint2d & ap1, 
-				const GeomPoint2d & ap2,
-				const GeomPoint2d & ap3)
-  : p1(ap1), p2(ap2), p3(ap3)
-{
-  Vec<2> v1,v2;
+  CircleSegment :: CircleSegment (const GeomPoint2d & ap1, 
+				  const GeomPoint2d & ap2,
+				  const GeomPoint2d & ap3)
+    : p1(ap1), p2(ap2), p3(ap3)
+  {
+    Vec<2> v1,v2;
   
-  v1 = p1 - p2;
-  v2 = p3 - p2;
+    v1 = p1 - p2;
+    v2 = p3 - p2;
   
-  Point<2> p1t(p1(0)+v1[1], p1(1)-v1[0]);
-  Point<2> p2t(p3(0)+v2[1], p3(1)-v2[0]);
-  Line2d g1t(p1, p1t), g2t(p3, p2t);
+    Point<2> p1t(p1(0)+v1[1], p1(1)-v1[0]);
+    Point<2> p2t(p3(0)+v2[1], p3(1)-v2[0]);
+    Line2d g1t(p1, p1t), g2t(p3, p2t);
   
-  pm 	  = CrossPoint (g1t,g2t);
-  radius  = Dist(pm,StartPI());
-  w1      = Angle(Vec2d (p1 - pm));
-  w3      = Angle(Vec2d (p3 - pm));
-  if ( fabs(w3-w1) > M_PI )
-    {  
-      if ( w3>M_PI )   w3 -= 2*M_PI;
-      if ( w1>M_PI )   w1 -= 2*M_PI;
-    }
-}
+    pm 	  = CrossPoint (g1t,g2t);
+    radius  = Dist(pm,StartPI());
+    w1      = Angle(Vec2d (p1 - pm));
+    w3      = Angle(Vec2d (p3 - pm));
+    if ( fabs(w3-w1) > M_PI )
+      {  
+	if ( w3>M_PI )   w3 -= 2*M_PI;
+	if ( w1>M_PI )   w1 -= 2*M_PI;
+      }
+  }
  
-Point<2>  CircleSegment :: GetPoint (double t) const
-{
-  if (t >= 1.0)  { return p3; }
+  Point<2>  CircleSegment :: GetPoint (double t) const
+  {
+    if (t >= 1.0)  { return p3; }
      
-  double phi = StartAngle() + t*(EndAngle()-StartAngle());
-  Vec2d  tmp(cos(phi),sin(phi));
+    double phi = StartAngle() + t*(EndAngle()-StartAngle());
+    Vec2d  tmp(cos(phi),sin(phi));
      
-  return pm + Radius()*tmp;
-}
+    return pm + Radius()*tmp;
+  }
   
-void CircleSegment :: PrintCoeff (ostream & ost) const
-{
-  double a,b,c,d,e,f;
+  void CircleSegment :: PrintCoeff (ostream & ost) const
+  {
+    double a,b,c,d,e,f;
  
-  a = b = 1.0;
-  c = 0.0;
-  d = -2.0 * pm[0];
-  e = -2.0 * pm[1];
-  f = sqr(pm[0]) + sqr(pm[1]) - sqr(Radius());
+    a = b = 1.0;
+    c = 0.0;
+    d = -2.0 * pm[0];
+    e = -2.0 * pm[1];
+    f = sqr(pm[0]) + sqr(pm[1]) - sqr(Radius());
  
-  ost << a << "  " << b << "  " << c << "  " << d << "  " << e << "  " << f ;
-  ost << endl;
-}
+    ost << a << "  " << b << "  " << c << "  " << d << "  " << e << "  " << f ;
+    ost << endl;
+  }
 
 
 
+  DiscretePointsSegment ::   DiscretePointsSegment (const ARRAY<Point<2> > & apts)
+    : pts (apts),
+      p1 (apts[0](0), apts[0](1), 1),
+      p2 (apts.Last()(0), apts.Last()(1), 1)
 
+  { ; }
 
 
-//########################################################################
+  DiscretePointsSegment :: ~DiscretePointsSegment ()
+  { ; }
 
+  Point<2> DiscretePointsSegment :: GetPoint (double t) const
+  {
+    double t1 = t * (pts.Size()-1);
+    int segnr = int(t1);
+    if (segnr < 0) segnr = 0;
+    if (segnr >= pts.Size()) segnr = pts.Size()-1;
 
+    double rest = t1 - segnr;
+    
+    return Point<2> ((1-rest)*pts[segnr](0) + rest*pts[segnr+1](0),
+		     (1-rest)*pts[segnr](1) + rest*pts[segnr+1](1));
+  }
 
+  
 
-void CalcPartition (double l, double h, double r1, double r2,
-		    double ra, double elto0, ARRAY<double> & points)
-{
-  int i, j, n, nel;
-  double sum, t, dt, fun, fperel, oldf, f;
-
-  n = 1000;
-
-  points.SetSize (0);
-
-  sum = 0;
-  dt = l / n;
-  t = 0.5 * dt;
-  for (i = 1; i <= n; i++)
-    {
-      fun = min3 (h/ra, t/elto0 + h/r1, (l-t)/elto0 + h/r2);
-      sum += dt / fun;
-      t += dt;
-    }
-
-  nel = int (sum+1);
-  fperel = sum / nel;
-
-  points.Append (0);
-
-  i = 1;
-  oldf = 0;
-  t = 0.5 * dt;
-  for (j = 1; j <= n && i < nel; j++)
-    {
-      fun = min3 (h/ra, t/elto0 + h/r1, (l-t)/elto0 + h/r2);
-
-      f = oldf + dt / fun;
-
-      while (f > i * fperel && i < nel)
-	{
-	  points.Append ( (l/n) * (j-1 +  (i * fperel - oldf) / (f - oldf)) );
-	  i++;
-	}
-      oldf = f;
-      t += dt;
-    }
-  points.Append (l);
-}
+
+
+
+  //########################################################################
+
+
+
+
+  void CalcPartition (double l, double h, double r1, double r2,
+		      double ra, double elto0, ARRAY<double> & points)
+  {
+    int i, j, n, nel;
+    double sum, t, dt, fun, fperel, oldf, f;
+
+    n = 1000;
+
+    points.SetSize (0);
+
+    sum = 0;
+    dt = l / n;
+    t = 0.5 * dt;
+    for (i = 1; i <= n; i++)
+      {
+	fun = min3 (h/ra, t/elto0 + h/r1, (l-t)/elto0 + h/r2);
+	sum += dt / fun;
+	t += dt;
+      }
+
+    nel = int (sum+1);
+    fperel = sum / nel;
+
+    points.Append (0);
+
+    i = 1;
+    oldf = 0;
+    t = 0.5 * dt;
+    for (j = 1; j <= n && i < nel; j++)
+      {
+	fun = min3 (h/ra, t/elto0 + h/r1, (l-t)/elto0 + h/r2);
+
+	f = oldf + dt / fun;
+
+	while (f > i * fperel && i < nel)
+	  {
+	    points.Append ( (l/n) * (j-1 +  (i * fperel - oldf) / (f - oldf)) );
+	    i++;
+	  }
+	oldf = f;
+	t += dt;
+      }
+    points.Append (l);
+  }
 
 
 }
diff --git a/Netgen/libsrc/geom2d/spline2d.hpp b/Netgen/libsrc/geom2d/spline2d.hpp
index bdd33d736b..f107a137f1 100644
--- a/Netgen/libsrc/geom2d/spline2d.hpp
+++ b/Netgen/libsrc/geom2d/spline2d.hpp
@@ -19,6 +19,7 @@ class GeomPoint2d : public Point<2>
 public:
   /// refinement to point
   double refatpoint;
+  bool hpref;
 
   GeomPoint2d ()
   { ; }
@@ -44,14 +45,16 @@ public:
   int bc;
   /// copy spline mesh from other spline (-1.. do not copy)
   int copyfrom;
-
+  /// perfrom anisotropic refinement (hp-refinement) to edge
+  bool hpref_left;
+  bool hpref_right;
   /// calculates length of curve
   virtual double Length () const;
   /// returns point at curve, 0 <= t <= 1
   virtual Point<2> GetPoint (double t) const = 0;
   /// partitionizes curve
   void Partition (double h, double elto0,
-		  Mesh & mesh, int segnr) const;
+		  Mesh & mesh, Point3dTree & searchtree, int segnr) const;
   /// returns initial point on curve
   virtual const GeomPoint2d & StartPI () const = 0;
   /// returns terminal point on curve
@@ -64,6 +67,8 @@ public:
   virtual void PrintCoeff (ostream & ost) const = 0;
 
   virtual void GetPoints (int n, ARRAY<Point<2> > & points);
+
+  virtual string GetType(void) const {return "splinebase";}
 };
 
 
@@ -85,6 +90,8 @@ public:
   virtual const GeomPoint2d & EndPI () const { return p2; }
   ///
   virtual void PrintCoeff (ostream & ost) const;
+
+  virtual string GetType(void) const {return "line";}
 };
 
 
@@ -106,6 +113,8 @@ public:
   virtual const GeomPoint2d & EndPI () const { return p3; }
   ///
   virtual void PrintCoeff (ostream & ost) const;
+
+  virtual string GetType(void) const {return "spline3";}
 };
 
 
@@ -137,6 +146,8 @@ public:
   double StartAngle() const { return w1; }
   ///
   double EndAngle() const { return w3; }
+
+  virtual string GetType(void) const {return "circle";}
 };
 
 
@@ -164,4 +175,30 @@ public:
 
 
 
+
+
+/// 
+class DiscretePointsSegment : public SplineSegment
+{
+  ARRAY<Point<2> > pts;
+  GeomPoint2d p1, p2;
+public:
+  ///
+  DiscretePointsSegment (const ARRAY<Point<2> > & apts);
+  ///
+  virtual ~DiscretePointsSegment ();
+  ///
+  virtual Point<2> GetPoint (double t) const;
+  ///
+  virtual const GeomPoint2d & StartPI () const { return p1; };
+  ///
+  virtual const GeomPoint2d & EndPI () const { return p2; }
+  ///
+  virtual void PrintCoeff (ostream & /* ost */) const { ; }
+};
+
+
+
+
+
 #endif
diff --git a/Netgen/libsrc/geom2d/splinegeometry2.cpp b/Netgen/libsrc/geom2d/splinegeometry2.cpp
index 921e22a295..a034d53989 100644
--- a/Netgen/libsrc/geom2d/splinegeometry2.cpp
+++ b/Netgen/libsrc/geom2d/splinegeometry2.cpp
@@ -19,62 +19,107 @@ namespace netgen
 
 
 
-
+SplineGeometry2d :: ~SplineGeometry2d()
+{
+  for(int i=0; i<splines.Size(); i++)
+    {
+      delete splines[i];
+    }
+  splines.DeleteAll();
+  geompoints.DeleteAll();
+}
 
 void SplineGeometry2d :: Load (const char * filename)
 {
   ifstream infile;
-  int nump, numseg, nelp, i, leftdom, rightdom;
+  int nump, numseg, leftdom, rightdom;
   double x, y;
   int hi1, hi2, hi3;
   double hd;
-  char reco[50], ch;
+  char buf[50], ch;
 
   infile.open (filename);
 
-  infile >> reco;
+  if (! infile.good() )
+    throw NgException(string ("2D Input file '") + 
+		      string (filename) +
+		      string ("' not available!"));
+
+  infile >> buf;   // file recognition
   infile >> elto0;
 
   infile >> nump;
-  for (i = 0; i < nump; i++)
+  for (int i = 0; i < nump; i++)
     {
       infile >> x >> y >> hd;
+
+      Flags flags;
+
+      ch = 'a';
+      // infile >> ch;
+      do {
+	infile.get (ch);
+      } while (isspace(ch) && ch != '\n');
+      while (ch == '-')
+	{
+	  char flag[100];
+	  flag[0]='-';
+	  infile >> (flag+1);
+	  flags.SetCommandLineFlag (flag);
+	  ch = 'a';
+	  do {
+	    infile.get (ch);
+	  } while (isspace(ch) && ch != '\n');
+	}
+    
+      if (infile.good())
+	infile.putback (ch);
+
       geompoints.Append (GeomPoint2d(x, y, hd));
+      geompoints.Last().hpref = flags.GetDefineFlag ("hpref");
     }
 
   infile >> numseg;
-
   SplineSegment * spline = 0;
-  for (i = 0; i < numseg; i++)
+  for (int i = 0; i < numseg; i++)
     {
       infile >> leftdom >> rightdom;
 
-      infile >> nelp;  
-      switch (nelp)
-	{ // type of spline segement
-	case 2:
-	  { // a line
-	    infile >> hi1 >> hi2;
-	    spline = new LineSegment(geompoints[hi1-1],
-				     geompoints[hi2-1]);
-	    break;
-	  }
-	case 3:
-	  { // a rational spline
-	    infile >> hi1 >> hi2 >> hi3;
-	    spline = new SplineSegment3 (geompoints[hi1-1],
-					 geompoints[hi2-1],
-					 geompoints[hi3-1]);
-	    break;
-	  }
-	case 4:
-	  { // an arc
-	    infile >> hi1 >> hi2 >> hi3;
-	    splines.Append (new CircleSegment (geompoints[hi1-1],
-					       geompoints[hi2-1],
-					       geompoints[hi3-1]));
-	    break;
-	  }
+      // cout << "add spline " << i << ", left = " << leftdom << endl;
+
+      infile >> buf;
+      // type of spline segement
+      if (strcmp (buf, "2") == 0)
+	{ // a line
+	  infile >> hi1 >> hi2;
+	  spline = new LineSegment(geompoints[hi1-1],
+				   geompoints[hi2-1]);
+	}
+      else if (strcmp (buf, "3") == 0)
+	{ // a rational spline
+	  infile >> hi1 >> hi2 >> hi3;
+	  spline = new SplineSegment3 (geompoints[hi1-1],
+				       geompoints[hi2-1],
+				       geompoints[hi3-1]);
+	}
+      else if (strcmp (buf, "4") == 0)
+	{ // an arc
+	  infile >> hi1 >> hi2 >> hi3;
+	  spline = new CircleSegment (geompoints[hi1-1],
+				      geompoints[hi2-1],
+				      geompoints[hi3-1]);
+	  break;
+	}
+      else if (strcmp (buf, "discretepoints") == 0)
+	{
+	  int npts;
+	  infile >> npts;
+	  ARRAY<Point<2> > pts(npts);
+	  for (int j = 0; j < npts; j++)
+	    infile >> pts[j](0) >> pts[j](1);
+
+	  spline = new DiscretePointsSegment (pts);
+	  cout << "pts = " << pts << endl;
 	}
     
       infile >> spline->reffak;
@@ -82,7 +127,6 @@ void SplineGeometry2d :: Load (const char * filename)
       spline -> rightdom = rightdom;
       splines.Append (spline);
 
-    
 
       Flags flags;
       ch = 'a';
@@ -101,13 +145,15 @@ void SplineGeometry2d :: Load (const char * filename)
 	infile.putback (ch);
     
       splines.Last()->bc = int (flags.GetNumFlag ("bc", i+1));
+      splines.Last()->hpref_left = int (flags.GetDefineFlag ("hpref")) || 
+	int (flags.GetDefineFlag ("hprefleft"));
+      splines.Last()->hpref_right = int (flags.GetDefineFlag ("hpref")) || 
+	int (flags.GetDefineFlag ("hprefright"));
       splines.Last()->copyfrom = int (flags.GetNumFlag ("copy", -1));
     }
 
 
-  Box<2> bbox;
-  GetBoundingBox (bbox);
-  // cout << "bbox = " << bbox << endl;
+  infile.close();
 }
 
 
@@ -115,20 +161,29 @@ void SplineGeometry2d :: Load (const char * filename)
 void SplineGeometry2d :: 
 PartitionBoundary (double h, Mesh & mesh2d)
 {
+  Box<2> bbox;
+  GetBoundingBox (bbox);
+  double dist = Dist (bbox.PMin(), bbox.PMax());
+  Point<3> pmin(bbox.PMin()(0), bbox.PMin()(1), -dist);
+  Point<3> pmax(bbox.PMax()(0), bbox.PMax()(1), dist);
+
+  cout << "searchtree from " << pmin << " to " << pmax << endl;
+  Point3dTree searchtree (pmin, pmax);
+  
   for (int i = 0; i < splines.Size(); i++)
     if (splines[i]->copyfrom == -1)
-      splines[i]->Partition(h, elto0, mesh2d, i+1);
+      splines[i]->Partition(h, elto0, mesh2d, searchtree, i+1);
     else
-      CopyEdgeMesh (splines[i]->copyfrom, i+1, mesh2d);
+      CopyEdgeMesh (splines[i]->copyfrom, i+1, mesh2d, searchtree);
 }
 
 
-void SplineGeometry2d :: CopyEdgeMesh (int from, int to, Mesh & mesh)
+void SplineGeometry2d :: CopyEdgeMesh (int from, int to, Mesh & mesh, Point3dTree & searchtree)
 {
   int i, j, k;
 
-  ARRAY<int> mappoints (mesh.GetNP());
-  ARRAY<double> param (mesh.GetNP());
+  ARRAY<int, PointIndex::BASE> mappoints (mesh.GetNP());
+  ARRAY<double, PointIndex::BASE> param (mesh.GetNP());
   mappoints = -1;
   param = 0;
 
@@ -168,8 +223,9 @@ void SplineGeometry2d :: CopyEdgeMesh (int from, int to, Mesh & mesh)
 	  if (npi == -1)
 	    {
 	      npi = mesh.AddPoint (newp3);
+	      searchtree.Insert (newp3, npi);
 	    }
-	  
+
 	  mappoints.Elem(i) = npi;
 
 	  mesh.GetIdentifications().Add (i, npi, to);
@@ -221,4 +277,70 @@ GetBoundingBox (Box<2> & box) const
     }
 }
 
+void SplineGeometry2d :: 
+SetGrading (const double grading)
+{ elto0 = grading;}
+
+void SplineGeometry2d :: 
+AppendPoint (const double x, const double y, const double reffac, const bool hpref)
+{
+  geompoints.Append (GeomPoint2d(x, y, reffac));
+  geompoints.Last().hpref = hpref;
+}
+
+
+void SplineGeometry2d :: 
+AppendSegment(SplineSegment * spline, const int leftdomain, const int rightdomain, 
+	      const int bc, 
+	      const double reffac, const bool hprefleft, const bool hprefright,
+	      const int copyfrom)
+{
+  spline -> leftdom = leftdomain;
+  spline -> rightdom = rightdomain;
+  spline -> bc = (bc >= 0) ? bc : (splines.Size()+1);
+  spline -> reffak = reffac;
+  spline -> hpref_left = hprefleft;
+  spline -> hpref_right = hprefright;
+  spline -> copyfrom = copyfrom;
+  
+  splines.Append(spline);
+}
+
+void SplineGeometry2d :: 
+AppendLineSegment (const int n1, const int n2, const int leftdomain, const int rightdomain,
+		   const int bc, 
+		   const double reffac, const bool hprefleft, const bool hprefright,
+		   const int copyfrom)
+{
+  SplineSegment * spline = new LineSegment(geompoints[n1],geompoints[n2]);
+  AppendSegment(spline,leftdomain,rightdomain,bc,reffac,hprefleft,hprefright,copyfrom);  
+}
+void SplineGeometry2d :: 
+AppendSplineSegment (const int n1, const int n2, const int n3, const int leftdomain, const int rightdomain, 
+		     const int bc,
+		     const double reffac, const bool hprefleft, const bool hprefright,
+		     const int copyfrom)
+{
+  SplineSegment * spline = new SplineSegment3(geompoints[n1],geompoints[n2],geompoints[n3]);
+  AppendSegment(spline,leftdomain,rightdomain,bc,reffac,hprefleft,hprefright,copyfrom);
+}
+void SplineGeometry2d :: 
+AppendCircleSegment (const int n1, const int n2, const int n3, const int leftdomain, const int rightdomain,
+		     const int bc,  
+		     const double reffac, const bool hprefleft, const bool hprefright,
+		     const int copyfrom)
+{
+  SplineSegment * spline = new CircleSegment(geompoints[n1],geompoints[n2],geompoints[n3]);
+  AppendSegment(spline,leftdomain,rightdomain,bc,reffac,hprefleft,hprefright,copyfrom);
+}
+void SplineGeometry2d :: 
+AppendDiscretePointsSegment (const ARRAY< Point<2> > & points, const int leftdomain, const int rightdomain, 
+			     const int bc, 
+			     const double reffac, const bool hprefleft, const bool hprefright,
+			     const int copyfrom)
+{
+  SplineSegment * spline = new DiscretePointsSegment(points);
+  AppendSegment(spline,leftdomain,rightdomain,bc,reffac,hprefleft,hprefright,copyfrom);
+}
+
 }
diff --git a/Netgen/libsrc/geom2d/splinegeometry2.hpp b/Netgen/libsrc/geom2d/splinegeometry2.hpp
index 38de574b24..66c2a4d79f 100644
--- a/Netgen/libsrc/geom2d/splinegeometry2.hpp
+++ b/Netgen/libsrc/geom2d/splinegeometry2.hpp
@@ -19,25 +19,65 @@ extern void PartitionBoundary (const ARRAY<SplineSegment*> & splines,
 			       double h, double elto0,
 			       Mesh & mesh2d);
 
-
 class SplineGeometry2d
 {
   ARRAY<GeomPoint2d> geompoints;
   ARRAY<SplineSegment*> splines;
   double elto0;
 
+
+private:
+  void AppendSegment(SplineSegment * spline, const int leftdomain, const int rightdomain,
+		     const int bc,
+		     const double reffac, const bool hprefleft, const bool hprefright,
+		     const int copyfrom);
+
 public:
+  ~SplineGeometry2d();
+
   void Load (const char * filename);
   void PartitionBoundary (double h, Mesh & mesh2d);
 
-  void CopyEdgeMesh (int from, int to, Mesh & mesh2d);
+  void CopyEdgeMesh (int from, int to, Mesh & mesh2d, Point3dTree & searchtree);
 
   const ARRAY<SplineSegment*> & GetSplines () const
   { return splines; }
 
   void GetBoundingBox (Box<2> & box) const;
+
+  int GetNP () const { return geompoints.Size(); }
+  const GeomPoint2d & GetPoint(int i) const { return geompoints[i]; }
+
+  void SetGrading (const double grading);
+  void AppendPoint (const double x, const double y, const double reffac = 1., const bool hpref = false);
+  
+  void AppendLineSegment (const int n1, const int n2,
+			  const int leftdomain, const int rightdomain, const int bc = -1,
+			  const double reffac = 1.,
+			  const bool hprefleft = false, const bool hprefright = false,
+			  const int copyfrom = -1);
+  void AppendSplineSegment (const int n1, const int n2, const int n3,
+			    const int leftdomain, const int rightdomain, const int bc = -1,
+			    const double reffac = 1.,
+			    const bool hprefleft = false, const bool hprefright = false,
+			    const int copyfrom = -1);
+  void AppendCircleSegment (const int n1, const int n2, const int n3,
+			    const int leftdomain, const int rightdomain, const int bc = -1,
+			    const double reffac = 1.,
+			    const bool hprefleft = false, const bool hprefright = false,
+			    const int copyfrom = -1);
+  void AppendDiscretePointsSegment (const ARRAY< Point<2> > & points, 
+				    const int leftdomain, const int rightdomain, const int bc = -1,
+				    const double reffac = 1.,
+				    const bool hprefleft = false, const bool hprefright = false,
+				    const int copyfrom = -1);
+  
 };
 
 
+void MeshFromSpline2D (SplineGeometry2d & geometry,
+		       Mesh *& mesh, 
+		       MeshingParameters & mp);
+
 
 #endif
diff --git a/Netgen/libsrc/gprim/adtree.cpp b/Netgen/libsrc/gprim/adtree.cpp
index b3e3f13795..25f46183e2 100644
--- a/Netgen/libsrc/gprim/adtree.cpp
+++ b/Netgen/libsrc/gprim/adtree.cpp
@@ -9,478 +9,461 @@ namespace netgen
 {
 
 
-/* ******************************* ADTree ******************************* */
-
-
-ADTreeNode :: ADTreeNode(int adim)
-{
-  pi = 0;
-
-  left = NULL;
-  right = NULL;
-  father = NULL;
-  nchilds = 0;
-  dim = adim;
-  data = new float [dim];
-  boxmin = NULL;
-  boxmax = NULL;
-}
-
-
-
-
-ADTreeNode :: ~ADTreeNode()
-{
-  delete data;
-}
-
-
-ADTree :: ADTree (int adim, const float * acmin, 
-			 const float * acmax)
-  : ela(0), stack(1000), stackdir(1000)
-{
-  dim = adim;
-  cmin = new float [dim];
-  cmax = new float [dim];
-  memcpy (cmin, acmin, dim * sizeof(float));
-  memcpy (cmax, acmax, dim * sizeof(float));
-
-  root = new ADTreeNode (dim);
-  root->sep = (cmin[0] + cmax[0]) / 2;
-  root->boxmin = new float [dim];
-  root->boxmax = new float [dim];
-  memcpy (root->boxmin, cmin, dim * sizeof(float));
-  memcpy (root->boxmax, cmax, dim * sizeof(float));
-}
-
-ADTree :: ~ADTree ()
-{
-  ;
-}
-
-void ADTree :: Insert (const float * p, int pi)
-{
-  ADTreeNode *node;
-  ADTreeNode *next;
-  int dir;
-  int lr;
-
-  float * bmin = new float [dim];
-  float * bmax = new float [dim];
+  /* ******************************* ADTree ******************************* */
+
+
+  ADTreeNode :: ADTreeNode(int adim)
+  {
+    pi = -1;
+
+    left = NULL;
+    right = NULL;
+    father = NULL;
+    nchilds = 0;
+    dim = adim;
+    data = new float [dim];
+    boxmin = NULL;
+    boxmax = NULL;
+  }
+
+
+
+
+  ADTreeNode :: ~ADTreeNode()
+  {
+    delete data;
+  }
+
+
+  ADTree :: ADTree (int adim, const float * acmin, 
+		    const float * acmax)
+    : ela(0), stack(1000), stackdir(1000)
+  {
+    dim = adim;
+    cmin = new float [dim];
+    cmax = new float [dim];
+    memcpy (cmin, acmin, dim * sizeof(float));
+    memcpy (cmax, acmax, dim * sizeof(float));
+
+    root = new ADTreeNode (dim);
+    root->sep = (cmin[0] + cmax[0]) / 2;
+    root->boxmin = new float [dim];
+    root->boxmax = new float [dim];
+    memcpy (root->boxmin, cmin, dim * sizeof(float));
+    memcpy (root->boxmax, cmax, dim * sizeof(float));
+  }
+
+  ADTree :: ~ADTree ()
+  {
+    ;
+  }
+
+  void ADTree :: Insert (const float * p, int pi)
+  {
+    ADTreeNode *node;
+    ADTreeNode *next;
+    int dir;
+    int lr;
+
+    float * bmin = new float [dim];
+    float * bmax = new float [dim];
   
-  memcpy (bmin, cmin, dim * sizeof(float));
-  memcpy (bmax, cmax, dim * sizeof(float));
+    memcpy (bmin, cmin, dim * sizeof(float));
+    memcpy (bmax, cmax, dim * sizeof(float));
 
-#ifdef MARK
-  MARK (insertloop3)
-#endif
-
-  next = root;
-  dir = 0;
-  while (next)
-    {
-      node = next;
 
-      if (!node->pi)
-	{    
-	  memcpy (node->data, p, dim * sizeof(float));
-	  node->pi = pi;
-
-	  if (ela.Size() < pi)
-	    ela.SetSize (pi);
-	  ela.Elem(pi) = node;
+    next = root;
+    dir = 0;
+    while (next)
+      {
+	node = next;
 
-	  return;
-	}
+	if (node->pi == -1)
+	  {    
+	    memcpy (node->data, p, dim * sizeof(float));
+	    node->pi = pi;
 
-      if (node->sep > p[dir])
-	{
-	  next = node->left;
-	  bmax[dir] = node->sep;
-	  lr = 0;
-	}
-      else
-	{
-	  next = node->right;
-	  bmin[dir] = node->sep;
-	  lr = 1;
-	}
+	    if (ela.Size() < pi+1)
+	      ela.SetSize (pi+1);
+	    ela[pi] = node;
 
-      dir++;
-      if (dir == dim)
-	dir = 0;
-    }
+	    return;
+	  }
 
-#ifdef MARK
-  MARK (insertend3)
-#endif
+	if (node->sep > p[dir])
+	  {
+	    next = node->left;
+	    bmax[dir] = node->sep;
+	    lr = 0;
+	  }
+	else
+	  {
+	    next = node->right;
+	    bmin[dir] = node->sep;
+	    lr = 1;
+	  }
 
+	dir++;
+	if (dir == dim)
+	  dir = 0;
+      }
 
-  next = new ADTreeNode(dim);
-  memcpy (next->data, p, dim * sizeof(float));
-  next->pi = pi;
-  next->sep = (bmin[dir] + bmax[dir]) / 2;
-  next->boxmin = bmin;
-  next->boxmax = bmax;
 
-  if (ela.Size() < pi)
-    ela.SetSize (pi);
-  ela.Elem(pi) = next;
+    next = new ADTreeNode(dim);
+    memcpy (next->data, p, dim * sizeof(float));
+    next->pi = pi;
+    next->sep = (bmin[dir] + bmax[dir]) / 2;
+    next->boxmin = bmin;
+    next->boxmax = bmax;
 
+    if (ela.Size() < pi+1)
+      ela.SetSize (pi+1);
+    ela[pi] = next;
 
-  if (lr)
-    node->right = next;
-  else
-    node->left = next;
-  next -> father = node;
 
-  while (node)
-    {
-      node->nchilds++;
-      node = node->father;
-    }
-}
+    if (lr)
+      node->right = next;
+    else
+      node->left = next;
+    next -> father = node;
 
-void ADTree :: DeleteElement (int pi)
-{
-  ADTreeNode * node = ela.Get(pi);
+    while (node)
+      {
+	node->nchilds++;
+	node = node->father;
+      }
+  }
 
-  node->pi = 0;
+  void ADTree :: DeleteElement (int pi)
+  {
+    ADTreeNode * node = ela[pi];
 
-  node = node->father;
-  while (node)
-    {
-      node->nchilds--;
-      node = node->father;
-    }
-}
+    node->pi = -1;
 
+    node = node->father;
+    while (node)
+      {
+	node->nchilds--;
+	node = node->father;
+      }
+  }
 
-void ADTree :: SetCriterion (ADTreeCriterion & acriterion)
-{
-  criterion = & acriterion;
-}
 
+  void ADTree :: SetCriterion (ADTreeCriterion & acriterion)
+  {
+    criterion = & acriterion;
+  }
 
-void ADTree :: Reset ()
-{
-  stack.Elem(1) = root;
-  stackdir.Elem(1) = 0;
-  stackindex = 1;
-}
 
+  void ADTree :: Reset ()
+  {
+    stack.Elem(1) = root;
+    stackdir.Elem(1) = 0;
+    stackindex = 1;
+  }
 
-int ADTree:: Next ()
-{
-  ADTreeNode *node;
-  int dir;
 
-  if (stackindex == 0)
-    return 0;
+  int ADTree:: Next ()
+  {
+    ADTreeNode *node;
+    int dir;
 
-  do 
-    {
-      node = stack.Get(stackindex);
-      dir = stackdir.Get(stackindex);
-      stackindex --;
+    if (stackindex == 0)
+      return -1;
 
-      if (criterion -> Eval(node))
-	{
-	  int ndir = dir + 1;
-	  if (ndir == dim)
-	    ndir = 0;
+    do 
+      {
+	node = stack.Get(stackindex);
+	dir = stackdir.Get(stackindex);
+	stackindex --;
 
-	  if (node -> left && criterion -> Eval (node->left))
-	    {
-	      stackindex ++;
-	      stack.Elem(stackindex) = node -> left;
-	      stackdir.Elem(stackindex) = ndir;
-	    }
-	  if (node->right && criterion -> Eval (node -> right))
-	    {
-	      stackindex++;
-	      stack.Elem(stackindex) = node->right;
-	      stackdir.Elem(stackindex) = ndir;
-	    }
+	if (criterion -> Eval(node))
+	  {
+	    int ndir = dir + 1;
+	    if (ndir == dim)
+	      ndir = 0;
+
+	    if (node -> left && criterion -> Eval (node->left))
+	      {
+		stackindex ++;
+		stack.Elem(stackindex) = node -> left;
+		stackdir.Elem(stackindex) = ndir;
+	      }
+	    if (node->right && criterion -> Eval (node -> right))
+	      {
+		stackindex++;
+		stack.Elem(stackindex) = node->right;
+		stackdir.Elem(stackindex) = ndir;
+	      }
 	  
-	  if (node -> pi)
-	    return node->pi;
-	}
-    }
-  while (stackindex > 0);
+	    if (node -> pi != -1)
+	      return node->pi;
+	  }
+      }
+    while (stackindex > 0);
 
-  return 0;
-}
+    return -1;
+  }
 
 
-void ADTree :: GetMatch (ARRAY <int> & matches)
-{
-  int nodenr;
+  void ADTree :: GetMatch (ARRAY <int> & matches)
+  {
+    int nodenr;
 
-  Reset();
+    Reset();
 
-  while (nodenr = Next())
-    matches.Append (nodenr);
-}
+    while ( (nodenr = Next()) != -1)
+      matches.Append (nodenr);
+  }
 
 
-void ADTree :: PrintRec (ostream & ost, const ADTreeNode * node) const
-{
+  void ADTree :: PrintRec (ostream & ost, const ADTreeNode * node) const
+  {
   
-  if (node->data)
-    {
-      ost << node->pi << ": ";
-      ost << node->nchilds << " childs, ";
-      for (int i = 0; i < dim; i++)
-	ost << node->data[i] << " ";
-      ost << endl;
-    }
-  if (node->left)
-    {
-      ost << "l ";
-      PrintRec (ost, node->left);
-    }
-  if (node->right)
-    {
-      ost << "r ";
-      PrintRec (ost, node->right);
-    }
-}
-
-
-/* ******************************* ADTree3 ******************************* */
+    if (node->data)
+      {
+	ost << node->pi << ": ";
+	ost << node->nchilds << " childs, ";
+	for (int i = 0; i < dim; i++)
+	  ost << node->data[i] << " ";
+	ost << endl;
+      }
+    if (node->left)
+      {
+	ost << "l ";
+	PrintRec (ost, node->left);
+      }
+    if (node->right)
+      {
+	ost << "r ";
+	PrintRec (ost, node->right);
+      }
+  }
 
 
-ADTreeNode3 :: ADTreeNode3()
-{
-  pi = 0;
+  /* ******************************* ADTree3 ******************************* */
 
-  left = NULL;
-  right = NULL;
-  father = NULL;
-  nchilds = 0;
-}
-
-void ADTreeNode3 :: DeleteChilds ()
-{
-  if (left)
-    {
-      left->DeleteChilds();
-      delete left;
-      left = NULL;
-    }
-  if (right)
-    {
-      right->DeleteChilds();
-      delete right;
-      right = NULL;
-    }
-}
 
+  ADTreeNode3 :: ADTreeNode3()
+  {
+    pi = -1;
 
-BlockAllocator ADTreeNode3 :: ball(sizeof (ADTreeNode3));
+    left = NULL;
+    right = NULL;
+    father = NULL;
+    nchilds = 0;
+  }
 
-
-void * ADTreeNode3 :: operator new(size_t s)
-{
-  ADTreeNode3 * ap;
-  return ball.Alloc();
-}
-
-void ADTreeNode3 :: operator delete (void * p)
-{
-  ball.Free (p);
-  //  ::delete [] p;
-}
+  void ADTreeNode3 :: DeleteChilds ()
+  {
+    if (left)
+      {
+	left->DeleteChilds();
+	delete left;
+	left = NULL;
+      }
+    if (right)
+      {
+	right->DeleteChilds();
+	delete right;
+	right = NULL;
+      }
+  }
 
 
+  BlockAllocator ADTreeNode3 :: ball(sizeof (ADTreeNode3));
 
 
+  void * ADTreeNode3 :: operator new(size_t s)
+  {
+    return ball.Alloc();
+  }
 
+  void ADTreeNode3 :: operator delete (void * p)
+  {
+    ball.Free (p);
+  }
 
 
-ADTree3 :: ADTree3 (const float * acmin, 
-			 const float * acmax)
-  : ela(0)
-{
-  memcpy (cmin, acmin, 3 * sizeof(float));
-  memcpy (cmax, acmax, 3 * sizeof(float));
 
-  root = new ADTreeNode3;
-  root->sep = (cmin[0] + cmax[0]) / 2;
-}
 
-ADTree3 :: ~ADTree3 ()
-{
-  root->DeleteChilds();
-  delete root;
-}
 
 
-void ADTree3 :: Insert (const float * p, int pi)
-{
-  ADTreeNode3 *node;
-  ADTreeNode3 *next;
-  int dir;
-  int lr;
 
-  float bmin[3];
-  float bmax[3];
+  ADTree3 :: ADTree3 (const float * acmin, 
+		      const float * acmax)
+    : ela(0)
+  {
+    memcpy (cmin, acmin, 3 * sizeof(float));
+    memcpy (cmax, acmax, 3 * sizeof(float));
+
+    root = new ADTreeNode3;
+    root->sep = (cmin[0] + cmax[0]) / 2;
+  }
+
+  ADTree3 :: ~ADTree3 ()
+  {
+    root->DeleteChilds();
+    delete root;
+  }
+
+
+  void ADTree3 :: Insert (const float * p, int pi)
+  {
+    ADTreeNode3 *node;
+    ADTreeNode3 *next;
+    int dir;
+    int lr;
+
+    float bmin[3];
+    float bmax[3];
   
-  memcpy (bmin, cmin, 3 * sizeof(float));
-  memcpy (bmax, cmax, 3 * sizeof(float));
-
-#ifdef MARK
-  //  MARK (insertloop3)
-#endif
-
-  next = root;
-  dir = 0;
-  while (next)
-    {
-      node = next;
-
-      if (!node->pi)
-	{    
-	  memcpy (node->data, p, 3 * sizeof(float));
-	  node->pi = pi;
-
-	  if (ela.Size() < pi)
-	    ela.SetSize (pi);
-	  ela.Elem(pi) = node;
+    memcpy (bmin, cmin, 3 * sizeof(float));
+    memcpy (bmax, cmax, 3 * sizeof(float));
 
-	  return;
-	}
-
-      if (node->sep > p[dir])
-	{
-	  next = node->left;
-	  bmax[dir] = node->sep;
-	  lr = 0;
-	}
-      else
-	{
-	  next = node->right;
-	  bmin[dir] = node->sep;
-	  lr = 1;
-	}
-
-      dir++;
-      if (dir == 3)
-	dir = 0;
-    }
+    next = root;
+    dir = 0;
+    while (next)
+      {
+	node = next;
 
-#ifdef MARK
-  //  MARK (insertend3)
-#endif
+	if (node->pi == -1)
+	  {    
+	    memcpy (node->data, p, 3 * sizeof(float));
+	    node->pi = pi;
 
+	    if (ela.Size() < pi+1)
+	      ela.SetSize (pi+1);
+	    ela[pi] = node;
 
-  next = new ADTreeNode3;
-  memcpy (next->data, p, 3 * sizeof(float));
-  next->pi = pi;
-  next->sep = (bmin[dir] + bmax[dir]) / 2;
+	    return;
+	  }
 
+	if (node->sep > p[dir])
+	  {
+	    next = node->left;
+	    bmax[dir] = node->sep;
+	    lr = 0;
+	  }
+	else
+	  {
+	    next = node->right;
+	    bmin[dir] = node->sep;
+	    lr = 1;
+	  }
 
-  if (ela.Size() < pi)
-    ela.SetSize (pi);
-  ela.Elem(pi) = next;		
+	dir++;
+	if (dir == 3)
+	  dir = 0;
+      }
 
 
-  if (lr)
-    node->right = next;
-  else
-    node->left = next;
-  next -> father = node;
+    next = new ADTreeNode3;
+    memcpy (next->data, p, 3 * sizeof(float));
+    next->pi = pi;
+    next->sep = (bmin[dir] + bmax[dir]) / 2;
 
-  while (node)
-    {
-      node->nchilds++;
-      node = node->father;
-    }
-}
 
-void ADTree3 :: DeleteElement (int pi)
-{
-  ADTreeNode3 * node = ela.Get(pi);
+    if (ela.Size() < pi+1)
+      ela.SetSize (pi+1);
+    ela[pi] = next;		
 
-  node->pi = 0;
 
-  node = node->father;
-  while (node)
-    {
-      node->nchilds--;
-      node = node->father;
-    }
-}
+    if (lr)
+      node->right = next;
+    else
+      node->left = next;
+    next -> father = node;
 
-void ADTree3 :: GetIntersecting (const float * bmin, 
-				 const float * bmax,
-				 ARRAY<int> & pis) const
-{
-  static ARRAY<ADTreeNode3*> stack(1000);
-  static ARRAY<int> stackdir(1000);
-  ADTreeNode3 * node;
-  int dir, i, stacks;
+    while (node)
+      {
+	node->nchilds++;
+	node = node->father;
+      }
+  }
 
-  stack.SetSize (1000);
-  stackdir.SetSize(1000);
-  pis.SetSize(0);
+  void ADTree3 :: DeleteElement (int pi)
+  {
+    ADTreeNode3 * node = ela[pi];
 
-  stack.Elem(1) = root;
-  stackdir.Elem(1) = 0;
-  stacks = 1;
+    node->pi = -1;
 
-  while (stacks)
-    {
-      node = stack.Get(stacks);
-      dir = stackdir.Get(stacks); 
-      stacks--;
+    node = node->father;
+    while (node)
+      {
+	node->nchilds--;
+	node = node->father;
+      }
+  }
+
+  void ADTree3 :: GetIntersecting (const float * bmin, 
+				   const float * bmax,
+				   ARRAY<int> & pis) const
+  {
+    static ARRAY<ADTreeNode3*> stack(1000);
+    static ARRAY<int> stackdir(1000);
+    ADTreeNode3 * node;
+    int dir, stacks;
+
+    stack.SetSize (1000);
+    stackdir.SetSize(1000);
+    pis.SetSize(0);
+
+    stack.Elem(1) = root;
+    stackdir.Elem(1) = 0;
+    stacks = 1;
+
+    while (stacks)
+      {
+	node = stack.Get(stacks);
+	dir = stackdir.Get(stacks); 
+	stacks--;
 
-      if (node->pi)
-	{
-	  if (node->data[0] >= bmin[0] && node->data[0] <= bmax[0] &&
-	      node->data[1] >= bmin[1] && node->data[1] <= bmax[1] &&
-	      node->data[2] >= bmin[2] && node->data[2] <= bmax[2])
+	if (node->pi != -1)
+	  {
+	    if (node->data[0] >= bmin[0] && node->data[0] <= bmax[0] &&
+		node->data[1] >= bmin[1] && node->data[1] <= bmax[1] &&
+		node->data[2] >= bmin[2] && node->data[2] <= bmax[2])
 
-	    pis.Append (node->pi);
-	}
+	      pis.Append (node->pi);
+	  }
 
 
-      int ndir = dir+1;
-      if (ndir == 3)
-	ndir = 0;
+	int ndir = dir+1;
+	if (ndir == 3)
+	  ndir = 0;
 
-      if (node->left && bmin[dir] <= node->sep)
-	{
-	  stacks++;
-	  stack.Elem(stacks) = node->left;
-	  stackdir.Elem(stacks) = ndir;
-	}
-      if (node->right && bmax[dir] >= node->sep)
-	{
-	  stacks++;
-	  stack.Elem(stacks) = node->right;
-	  stackdir.Elem(stacks) = ndir;
-	}
-    }
-}
+	if (node->left && bmin[dir] <= node->sep)
+	  {
+	    stacks++;
+	    stack.Elem(stacks) = node->left;
+	    stackdir.Elem(stacks) = ndir;
+	  }
+	if (node->right && bmax[dir] >= node->sep)
+	  {
+	    stacks++;
+	    stack.Elem(stacks) = node->right;
+	    stackdir.Elem(stacks) = ndir;
+	  }
+      }
+  }
 
-void ADTree3 :: PrintRec (ostream & ost, const ADTreeNode3 * node) const
-{
+  void ADTree3 :: PrintRec (ostream & ost, const ADTreeNode3 * node) const
+  {
   
-  if (node->data)
-    {
-      ost << node->pi << ": ";
-      ost << node->nchilds << " childs, ";
-      for (int i = 0; i < 3; i++)
-	ost << node->data[i] << " ";
-      ost << endl;
-    }
-  if (node->left)
-    PrintRec (ost, node->left);
-  if (node->right)
-    PrintRec (ost, node->right);
-}
+    if (node->data)
+      {
+	ost << node->pi << ": ";
+	ost << node->nchilds << " childs, ";
+	for (int i = 0; i < 3; i++)
+	  ost << node->data[i] << " ";
+	ost << endl;
+      }
+    if (node->left)
+      PrintRec (ost, node->left);
+    if (node->right)
+      PrintRec (ost, node->right);
+  }
 
 
 
@@ -489,285 +472,277 @@ void ADTree3 :: PrintRec (ostream & ost, const ADTreeNode3 * node) const
 
 
 
+#ifdef ABC
 
-/* ******************************* ADTree3Div ******************************* */
+  /* ******************************* ADTree3Div ******************************* */
 
 
-ADTreeNode3Div :: ADTreeNode3Div()
-{
-  pi = 0;
+  ADTreeNode3Div :: ADTreeNode3Div()
+  {
+    pi = 0;
   
-  int i;
-  for (i = 0; i < ADTN_DIV; i++)
-    childs[i] = NULL;
-
-  father = NULL;
-  nchilds = 0;
-  minx = 0;
-  dist = 1;
-}
+    int i;
+    for (i = 0; i < ADTN_DIV; i++)
+      childs[i] = NULL;
 
-void ADTreeNode3Div :: DeleteChilds ()
-{
-  int i;
-  for (i = 0; i < ADTN_DIV; i++)
-    if (childs[i])
-      {
-	childs[i]->DeleteChilds();
-	delete childs[i];
-	childs[i] = NULL;
-      }
-}
+    father = NULL;
+    nchilds = 0;
+    minx = 0;
+    dist = 1;
+  }
 
+  void ADTreeNode3Div :: DeleteChilds ()
+  {
+    int i;
+    for (i = 0; i < ADTN_DIV; i++)
+      if (childs[i])
+	{
+	  childs[i]->DeleteChilds();
+	  delete childs[i];
+	  childs[i] = NULL;
+	}
+  }
 
-BlockAllocator ADTreeNode3Div :: ball(sizeof (ADTreeNode3Div));
 
-void * ADTreeNode3Div :: operator new(size_t)
-{
-  return ball.Alloc();
-}
+  BlockAllocator ADTreeNode3Div :: ball(sizeof (ADTreeNode3Div));
 
-void ADTreeNode3Div :: operator delete (void * p)
-{
-  ball.Free (p);
-}
+  void * ADTreeNode3Div :: operator new(size_t)
+  {
+    return ball.Alloc();
+  }
 
+  void ADTreeNode3Div :: operator delete (void * p)
+  {
+    ball.Free (p);
+  }
 
 
 
 
 
 
-ADTree3Div :: ADTree3Div (const float * acmin, 
-			  const float * acmax)
-  : ela(0)
-{
-  memcpy (cmin, acmin, 3 * sizeof(float));
-  memcpy (cmax, acmax, 3 * sizeof(float));
 
-  root = new ADTreeNode3Div;
+  ADTree3Div :: ADTree3Div (const float * acmin, 
+			    const float * acmax)
+    : ela(0)
+  {
+    memcpy (cmin, acmin, 3 * sizeof(float));
+    memcpy (cmax, acmax, 3 * sizeof(float));
 
-  root->minx = cmin[0];
-  root->dist = (cmax[0] - cmin[0]) / ADTN_DIV;
+    root = new ADTreeNode3Div;
 
-  //  root->sep = (cmin[0] + cmax[0]) / 2;
-}
+    root->minx = cmin[0];
+    root->dist = (cmax[0] - cmin[0]) / ADTN_DIV;
 
-ADTree3Div :: ~ADTree3Div ()
-{
-  root->DeleteChilds();
-  delete root;
-}
+    //  root->sep = (cmin[0] + cmax[0]) / 2;
+  }
 
+  ADTree3Div :: ~ADTree3Div ()
+  {
+    root->DeleteChilds();
+    delete root;
+  }
 
-void ADTree3Div :: Insert (const float * p, int pi)
-{
-  ADTreeNode3Div *node;
-  ADTreeNode3Div *next;
-  int dir;
-  int bag;
+
+  void ADTree3Div :: Insert (const float * p, int pi)
+  {
+    ADTreeNode3Div *node;
+    ADTreeNode3Div *next;
+    int dir;
+    int bag;
   
-  float bmin[3];
-  float bmax[3];
+    float bmin[3];
+    float bmax[3];
   
-  memcpy (bmin, cmin, 3 * sizeof(float));
-  memcpy (bmax, cmax, 3 * sizeof(float));
+    memcpy (bmin, cmin, 3 * sizeof(float));
+    memcpy (bmax, cmax, 3 * sizeof(float));
 
-#ifdef MARK
-  //  MARK (insertloop3)
-#endif
 
-  //  (*testout) << endl << "add point " << Point3d (p[0], p[1], p[2]) << endl;
-
-  next = root;
-  dir = 0;
-  while (next)
-    {
-      node = next;
+    next = root;
+    dir = 0;
+    while (next)
+      {
+	node = next;
 
-      if (!node->pi)
-	{    
-	  memcpy (node->data, p, 3 * sizeof(float));
-	  node->pi = pi;
+	if (!node->pi)
+	  {    
+	    memcpy (node->data, p, 3 * sizeof(float));
+	    node->pi = pi;
 
-	  if (ela.Size() < pi)
-	    ela.SetSize (pi);
-	  ela.Elem(pi) = node;
+	    if (ela.Size() < pi)
+	      ela.SetSize (pi);
+	    ela.Elem(pi) = node;
 
-	  return;
-	}
+	    return;
+	  }
 
-      double dx = (bmax[dir] - bmin[dir]) / ADTN_DIV;
-      bag = int ((p[dir]-bmin[dir]) / dx);
+	double dx = (bmax[dir] - bmin[dir]) / ADTN_DIV;
+	bag = int ((p[dir]-bmin[dir]) / dx);
 
-      //      (*testout) << "insert, bag = " << bag << endl;
+	//      (*testout) << "insert, bag = " << bag << endl;
 
-      if (bag < 0) bag = 0;
-      if (bag >= ADTN_DIV) bag = ADTN_DIV-1;
+	if (bag < 0) bag = 0;
+	if (bag >= ADTN_DIV) bag = ADTN_DIV-1;
       
-      double nbmin = bmin[dir] + bag * dx;
-      double nbmax = bmin[dir] + (bag+1) * dx;
+	double nbmin = bmin[dir] + bag * dx;
+	double nbmax = bmin[dir] + (bag+1) * dx;
 
-      /*      
-      (*testout) << "bmin, max = " << bmin[dir] << "-" << bmax[dir]
-      		 << " p = " << p[dir];
-      */
-      next = node->childs[bag];
-      bmin[dir] = nbmin;
-      bmax[dir] = nbmax;
+	/*      
+		(*testout) << "bmin, max = " << bmin[dir] << "-" << bmax[dir]
+		<< " p = " << p[dir];
+	*/
+	next = node->childs[bag];
+	bmin[dir] = nbmin;
+	bmax[dir] = nbmax;
 
-      //      (*testout) << "new bmin, max = " << bmin[dir] << "-" << bmax[dir] << endl;
+	//      (*testout) << "new bmin, max = " << bmin[dir] << "-" << bmax[dir] << endl;
 
       
-      /*      
-      if (node->sep > p[dir])
-	{
-	  next = node->left;
-	  bmax[dir] = node->sep;
-	  lr = 0;
-	}
-      else
-	{
-	  next = node->right;
-	  bmin[dir] = node->sep;
-	  lr = 1;
-	}
-      */
-
-      dir++;
-      if (dir == 3)
-	dir = 0;
-    }
-
-#ifdef MARK
-  //  MARK (insertend3)
-#endif
-
-
-  next = new ADTreeNode3Div;
-  memcpy (next->data, p, 3 * sizeof(float));
-  next->pi = pi;
-
-  next->minx = bmin[dir];
-  next->dist = (bmax[dir] - bmin[dir]) / ADTN_DIV;
-  //  next->sep = (bmin[dir] + bmax[dir]) / 2;
+	/*      
+		if (node->sep > p[dir])
+		{
+		next = node->left;
+		bmax[dir] = node->sep;
+		lr = 0;
+		}
+		else
+		{
+		next = node->right;
+		bmin[dir] = node->sep;
+		lr = 1;
+		}
+	*/
 
+	dir++;
+	if (dir == 3)
+	  dir = 0;
+      }
 
-  if (ela.Size() < pi)
-    ela.SetSize (pi);
-  ela.Elem(pi) = next;
 
-  node->childs[bag] = next;
-  next -> father = node;
+    next = new ADTreeNode3Div;
+    memcpy (next->data, p, 3 * sizeof(float));
+    next->pi = pi;
 
-  while (node)
-    {
-      node->nchilds++;
-      node = node->father;
-    }
-}
+    next->minx = bmin[dir];
+    next->dist = (bmax[dir] - bmin[dir]) / ADTN_DIV;
+    //  next->sep = (bmin[dir] + bmax[dir]) / 2;
 
-void ADTree3Div :: DeleteElement (int pi)
-{
-  ADTreeNode3Div * node = ela.Get(pi);
 
-  node->pi = 0;
+    if (ela.Size() < pi)
+      ela.SetSize (pi);
+    ela.Elem(pi) = next;
 
-  node = node->father;
-  while (node)
-    {
-      node->nchilds--;
-      node = node->father;
-    }
-}
+    node->childs[bag] = next;
+    next -> father = node;
 
-void ADTree3Div :: GetIntersecting (const float * bmin, 
-				    const float * bmax,
-				    ARRAY<int> & pis) const
-{
-  static ARRAY<ADTreeNode3Div*> stack(1000);
-  static ARRAY<int> stackdir(1000);
-  ADTreeNode3Div * node;
-  int dir, i, stacks;
+    while (node)
+      {
+	node->nchilds++;
+	node = node->father;
+      }
+  }
 
-  stack.SetSize (1000);
-  stackdir.SetSize(1000);
-  pis.SetSize(0);
+  void ADTree3Div :: DeleteElement (int pi)
+  {
+    ADTreeNode3Div * node = ela.Get(pi);
 
-  stack.Elem(1) = root;
-  stackdir.Elem(1) = 0;
-  stacks = 1;
+    node->pi = 0;
 
-  while (stacks)
-    {
-      node = stack.Get(stacks);
-      dir = stackdir.Get(stacks); 
-      stacks--;
+    node = node->father;
+    while (node)
+      {
+	node->nchilds--;
+	node = node->father;
+      }
+  }
+
+  void ADTree3Div :: GetIntersecting (const float * bmin, 
+				      const float * bmax,
+				      ARRAY<int> & pis) const
+  {
+    static ARRAY<ADTreeNode3Div*> stack(1000);
+    static ARRAY<int> stackdir(1000);
+    ADTreeNode3Div * node;
+    int dir, i, stacks;
+
+    stack.SetSize (1000);
+    stackdir.SetSize(1000);
+    pis.SetSize(0);
+
+    stack.Elem(1) = root;
+    stackdir.Elem(1) = 0;
+    stacks = 1;
+
+    while (stacks)
+      {
+	node = stack.Get(stacks);
+	dir = stackdir.Get(stacks); 
+	stacks--;
 
-      if (node->pi)
-	{
-	  if (node->data[0] >= bmin[0] && node->data[0] <= bmax[0] &&
-	      node->data[1] >= bmin[1] && node->data[1] <= bmax[1] &&
-	      node->data[2] >= bmin[2] && node->data[2] <= bmax[2])
+	if (node->pi)
+	  {
+	    if (node->data[0] >= bmin[0] && node->data[0] <= bmax[0] &&
+		node->data[1] >= bmin[1] && node->data[1] <= bmax[1] &&
+		node->data[2] >= bmin[2] && node->data[2] <= bmax[2])
 
-	    pis.Append (node->pi);
-	}
+	      pis.Append (node->pi);
+	  }
 
 
-      int ndir = dir+1;
-      if (ndir == 3)
-	ndir = 0;
+	int ndir = dir+1;
+	if (ndir == 3)
+	  ndir = 0;
 
-      int mini = int ( (bmin[dir] - node->minx) / node->dist );
-      int maxi = int ( (bmax[dir] - node->minx) / node->dist );
+	int mini = int ( (bmin[dir] - node->minx) / node->dist );
+	int maxi = int ( (bmax[dir] - node->minx) / node->dist );
       
-      //      (*testout) << "get int, mini, maxi = " << mini << ", " << maxi << endl;
-      if (mini < 0) mini = 0;
-      if (maxi >= ADTN_DIV) maxi = ADTN_DIV-1;
+	//      (*testout) << "get int, mini, maxi = " << mini << ", " << maxi << endl;
+	if (mini < 0) mini = 0;
+	if (maxi >= ADTN_DIV) maxi = ADTN_DIV-1;
 
-      for (i = mini; i <= maxi; i++)
-	if (node->childs[i])
-	  {
-	    stacks++;
-	    stack.Elem(stacks) = node->childs[i];
-	    stackdir.Elem(stacks) = ndir;
-	  }
+	for (i = mini; i <= maxi; i++)
+	  if (node->childs[i])
+	    {
+	      stacks++;
+	      stack.Elem(stacks) = node->childs[i];
+	      stackdir.Elem(stacks) = ndir;
+	    }
 
 
-      /*
-      if (node->left && bmin[dir] <= node->sep)
-	{
+	/*
+	  if (node->left && bmin[dir] <= node->sep)
+	  {
 	  stacks++;
 	  stack.Elem(stacks) = node->left;
 	  stackdir.Elem(stacks) = ndir;
-	}
-      if (node->right && bmax[dir] >= node->sep)
-	{
+	  }
+	  if (node->right && bmax[dir] >= node->sep)
+	  {
 	  stacks++;
 	  stack.Elem(stacks) = node->right;
 	  stackdir.Elem(stacks) = ndir;
-	}
-      */
-    }
-}
+	  }
+	*/
+      }
+  }
 
-void ADTree3Div :: PrintRec (ostream & ost, const ADTreeNode3Div * node) const
-{
+  void ADTree3Div :: PrintRec (ostream & ost, const ADTreeNode3Div * node) const
+  {
   
-  if (node->data)
-    {
-      ost << node->pi << ": ";
-      ost << node->nchilds << " childs, ";
-      ost << " from " << node->minx << " - " << node->minx + node->dist*ADTN_DIV << "  ";
-      for (int i = 0; i < 3; i++)
-	ost << node->data[i] << " ";
-      ost << endl;
-    }
-  int i;
-  for (i = 0; i < ADTN_DIV; i++)
-    if (node->childs[i])
-      PrintRec (ost, node->childs[i]);
-}
+    if (node->data)
+      {
+	ost << node->pi << ": ";
+	ost << node->nchilds << " childs, ";
+	ost << " from " << node->minx << " - " << node->minx + node->dist*ADTN_DIV << "  ";
+	for (int i = 0; i < 3; i++)
+	  ost << node->data[i] << " ";
+	ost << endl;
+      }
+    int i;
+    for (i = 0; i < ADTN_DIV; i++)
+      if (node->childs[i])
+	PrintRec (ost, node->childs[i]);
+  }
 
 
 
@@ -780,245 +755,237 @@ void ADTree3Div :: PrintRec (ostream & ost, const ADTreeNode3Div * node) const
 
 
 
-/* ******************************* ADTree3M ******************************* */
+  /* ******************************* ADTree3M ******************************* */
 
 
-ADTreeNode3M :: ADTreeNode3M()
-{
-  int i;
-  for (i = 0; i < ADTN_SIZE; i++)
-    pi[i] = 0;
-
-  left = NULL;
-  right = NULL;
-  father = NULL;
-  nchilds = 0;
-}
+  ADTreeNode3M :: ADTreeNode3M()
+  {
+    int i;
+    for (i = 0; i < ADTN_SIZE; i++)
+      pi[i] = 0;
 
-void ADTreeNode3M :: DeleteChilds ()
-{
-  if (left)
-    {
-      left->DeleteChilds();
-      delete left;
-      left = NULL;
-    }
-  if (right)
-    {
-      right->DeleteChilds();
-      delete right;
-      right = NULL;
-    }
-}
+    left = NULL;
+    right = NULL;
+    father = NULL;
+    nchilds = 0;
+  }
 
+  void ADTreeNode3M :: DeleteChilds ()
+  {
+    if (left)
+      {
+	left->DeleteChilds();
+	delete left;
+	left = NULL;
+      }
+    if (right)
+      {
+	right->DeleteChilds();
+	delete right;
+	right = NULL;
+      }
+  }
 
-BlockAllocator ADTreeNode3M :: ball(sizeof (ADTreeNode3M));
 
-void * ADTreeNode3M :: operator new(size_t)
-{
-  return ball.Alloc();
-}
+  BlockAllocator ADTreeNode3M :: ball(sizeof (ADTreeNode3M));
 
-void ADTreeNode3M :: operator delete (void * p)
-{
-  ball.Free (p);
-}
+  void * ADTreeNode3M :: operator new(size_t)
+  {
+    return ball.Alloc();
+  }
 
+  void ADTreeNode3M :: operator delete (void * p)
+  {
+    ball.Free (p);
+  }
 
 
 
 
 
 
-ADTree3M :: ADTree3M (const float * acmin, 
-			 const float * acmax)
-  : ela(0)
-{
-  memcpy (cmin, acmin, 3 * sizeof(float));
-  memcpy (cmax, acmax, 3 * sizeof(float));
 
-  root = new ADTreeNode3M;
-  root->sep = (cmin[0] + cmax[0]) / 2;
-}
+  ADTree3M :: ADTree3M (const float * acmin, 
+			const float * acmax)
+    : ela(0)
+  {
+    memcpy (cmin, acmin, 3 * sizeof(float));
+    memcpy (cmax, acmax, 3 * sizeof(float));
 
-ADTree3M :: ~ADTree3M ()
-{
-  root->DeleteChilds();
-  delete root;
-}
+    root = new ADTreeNode3M;
+    root->sep = (cmin[0] + cmax[0]) / 2;
+  }
 
+  ADTree3M :: ~ADTree3M ()
+  {
+    root->DeleteChilds();
+    delete root;
+  }
 
-void ADTree3M :: Insert (const float * p, int pi)
-{
-  ADTreeNode3M *node;
-  ADTreeNode3M *next;
-  int dir;
-  int lr;
-  int i;
-  float bmin[3];
-  float bmax[3];
-  
-  memcpy (bmin, cmin, 3 * sizeof(float));
-  memcpy (bmax, cmax, 3 * sizeof(float));
 
-#ifdef MARK
-  //  MARK (insertloop3)
-#endif
+  void ADTree3M :: Insert (const float * p, int pi)
+  {
+    ADTreeNode3M *node;
+    ADTreeNode3M *next;
+    int dir;
+    int lr;
+    int i;
+    float bmin[3];
+    float bmax[3];
+  
+    memcpy (bmin, cmin, 3 * sizeof(float));
+    memcpy (bmax, cmax, 3 * sizeof(float));
 
-  next = root;
-  dir = 0;
-  while (next)
-    {
-      node = next;
+    next = root;
+    dir = 0;
+    while (next)
+      {
+	node = next;
 
-      for (i = 0; i < ADTN_SIZE; i++)
-	if (!node->pi[i])
-	  {    
-	    memcpy (node->data[i], p, 3 * sizeof(float));
-	    node->pi[i] = pi;
+	for (i = 0; i < ADTN_SIZE; i++)
+	  if (!node->pi[i])
+	    {    
+	      memcpy (node->data[i], p, 3 * sizeof(float));
+	      node->pi[i] = pi;
 	    
-	    if (ela.Size() < pi)
-	      ela.SetSize (pi);
-	    ela.Elem(pi) = node;
+	      if (ela.Size() < pi)
+		ela.SetSize (pi);
+	      ela.Elem(pi) = node;
 	    
-	    return;
-	  }
-
-      if (node->sep > p[dir])
-	{
-	  next = node->left;
-	  bmax[dir] = node->sep;
-	  lr = 0;
-	}
-      else
-	{
-	  next = node->right;
-	  bmin[dir] = node->sep;
-	  lr = 1;
-	}
+	      return;
+	    }
 
-      dir++;
-      if (dir == 3)
-	dir = 0;
-    }
+	if (node->sep > p[dir])
+	  {
+	    next = node->left;
+	    bmax[dir] = node->sep;
+	    lr = 0;
+	  }
+	else
+	  {
+	    next = node->right;
+	    bmin[dir] = node->sep;
+	    lr = 1;
+	  }
 
-#ifdef MARK
-  //  MARK (insertend3)
-#endif
+	dir++;
+	if (dir == 3)
+	  dir = 0;
+      }
 
 
-  next = new ADTreeNode3M;
-  memcpy (next->data[0], p, 3 * sizeof(float));
-  next->pi[0] = pi;
-  next->sep = (bmin[dir] + bmax[dir]) / 2;
+    next = new ADTreeNode3M;
+    memcpy (next->data[0], p, 3 * sizeof(float));
+    next->pi[0] = pi;
+    next->sep = (bmin[dir] + bmax[dir]) / 2;
 
 
-  if (ela.Size() < pi)
-    ela.SetSize (pi);
-  ela.Elem(pi) = next;
+    if (ela.Size() < pi)
+      ela.SetSize (pi);
+    ela.Elem(pi) = next;
 
 
-  if (lr)
-    node->right = next;
-  else
-    node->left = next;
-  next -> father = node;
+    if (lr)
+      node->right = next;
+    else
+      node->left = next;
+    next -> father = node;
 
-  while (node)
-    {
-      node->nchilds++;
-      node = node->father;
-    }
-}
+    while (node)
+      {
+	node->nchilds++;
+	node = node->father;
+      }
+  }
 
-void ADTree3M :: DeleteElement (int pi)
-{
-  ADTreeNode3M * node = ela.Get(pi);
+  void ADTree3M :: DeleteElement (int pi)
+  {
+    ADTreeNode3M * node = ela.Get(pi);
 
-  int i;
-  for (i = 0; i < ADTN_SIZE; i++)
-    if (node->pi[i] == pi)
-      node->pi[i] = 0;
+    int i;
+    for (i = 0; i < ADTN_SIZE; i++)
+      if (node->pi[i] == pi)
+	node->pi[i] = 0;
 
-  node = node->father;
-  while (node)
-    {
-      node->nchilds--;
-      node = node->father;
-    }
-}
+    node = node->father;
+    while (node)
+      {
+	node->nchilds--;
+	node = node->father;
+      }
+  }
 
-void ADTree3M :: GetIntersecting (const float * bmin, 
-				 const float * bmax,
-				 ARRAY<int> & pis) const
-{
-  static ARRAY<ADTreeNode3M*> stack(1000);
-  static ARRAY<int> stackdir(1000);
-  ADTreeNode3M * node;
-  int dir, i, stacks;
+  void ADTree3M :: GetIntersecting (const float * bmin, 
+				    const float * bmax,
+				    ARRAY<int> & pis) const
+  {
+    static ARRAY<ADTreeNode3M*> stack(1000);
+    static ARRAY<int> stackdir(1000);
+    ADTreeNode3M * node;
+    int dir, i, stacks;
 
-  stack.SetSize (1000);
-  stackdir.SetSize(1000);
-  pis.SetSize(0);
+    stack.SetSize (1000);
+    stackdir.SetSize(1000);
+    pis.SetSize(0);
 
-  stack.Elem(1) = root;
-  stackdir.Elem(1) = 0;
-  stacks = 1;
+    stack.Elem(1) = root;
+    stackdir.Elem(1) = 0;
+    stacks = 1;
 
-  while (stacks)
-    {
-      node = stack.Get(stacks);
-      dir = stackdir.Get(stacks); 
-      stacks--;
+    while (stacks)
+      {
+	node = stack.Get(stacks);
+	dir = stackdir.Get(stacks); 
+	stacks--;
 
-      int * hpi = node->pi;
-      for (i = 0; i < ADTN_SIZE; i++)
-	if (hpi[i])
-	  {
-	    float * datai = &node->data[i][0];
-	    if (datai[0] >= bmin[0] && datai[0] <= bmax[0] &&
-		datai[1] >= bmin[1] && datai[1] <= bmax[1] &&
-		datai[2] >= bmin[2] && datai[2] <= bmax[2])
+	int * hpi = node->pi;
+	for (i = 0; i < ADTN_SIZE; i++)
+	  if (hpi[i])
+	    {
+	      float * datai = &node->data[i][0];
+	      if (datai[0] >= bmin[0] && datai[0] <= bmax[0] &&
+		  datai[1] >= bmin[1] && datai[1] <= bmax[1] &&
+		  datai[2] >= bmin[2] && datai[2] <= bmax[2])
 	      
-	      pis.Append (node->pi[i]);
-	  }
+		pis.Append (node->pi[i]);
+	    }
 
 
-      int ndir = dir+1;
-      if (ndir == 3)
-	ndir = 0;
+	int ndir = dir+1;
+	if (ndir == 3)
+	  ndir = 0;
 
-      if (node->left && bmin[dir] <= node->sep)
-	{
-	  stacks++;
-	  stack.Elem(stacks) = node->left;
-	  stackdir.Elem(stacks) = ndir;
-	}
-      if (node->right && bmax[dir] >= node->sep)
-	{
-	  stacks++;
-	  stack.Elem(stacks) = node->right;
-	  stackdir.Elem(stacks) = ndir;
-	}
-    }
-}
+	if (node->left && bmin[dir] <= node->sep)
+	  {
+	    stacks++;
+	    stack.Elem(stacks) = node->left;
+	    stackdir.Elem(stacks) = ndir;
+	  }
+	if (node->right && bmax[dir] >= node->sep)
+	  {
+	    stacks++;
+	    stack.Elem(stacks) = node->right;
+	    stackdir.Elem(stacks) = ndir;
+	  }
+      }
+  }
 
-void ADTree3M :: PrintRec (ostream & ost, const ADTreeNode3M * node) const
-{
+  void ADTree3M :: PrintRec (ostream & ost, const ADTreeNode3M * node) const
+  {
   
-  if (node->data)
-    {
-      //      ost << node->pi << ": ";
-      ost << node->nchilds << " childs, ";
-      for (int i = 0; i < 3; i++)
-	ost << node->data[i] << " ";
-      ost << endl;
-    }
-  if (node->left)
-    PrintRec (ost, node->left);
-  if (node->right)
-    PrintRec (ost, node->right);
-}
+    if (node->data)
+      {
+	//      ost << node->pi << ": ";
+	ost << node->nchilds << " childs, ";
+	for (int i = 0; i < 3; i++)
+	  ost << node->data[i] << " ";
+	ost << endl;
+      }
+    if (node->left)
+      PrintRec (ost, node->left);
+    if (node->right)
+      PrintRec (ost, node->right);
+  }
 
 
 
@@ -1031,44 +998,44 @@ void ADTree3M :: PrintRec (ostream & ost, const ADTreeNode3M * node) const
 
 
 
-/* ******************************* ADTree3F ******************************* */
+  /* ******************************* ADTree3F ******************************* */
 
 
-ADTreeNode3F :: ADTreeNode3F()
-{
-  pi = 0;
-  father = NULL;
-  nchilds = 0;
-  int i;
-  for (i = 0; i < 8; i++)
-    childs[i] = NULL;
-}
+  ADTreeNode3F :: ADTreeNode3F()
+  {
+    pi = 0;
+    father = NULL;
+    nchilds = 0;
+    int i;
+    for (i = 0; i < 8; i++)
+      childs[i] = NULL;
+  }
 
-void ADTreeNode3F :: DeleteChilds ()
-{
-  int i;
+  void ADTreeNode3F :: DeleteChilds ()
+  {
+    int i;
 
-  for (i = 0; i < 8; i++)
-    {
-      if (childs[i])
-	childs[i]->DeleteChilds();
-      delete childs[i];
-      childs[i] = NULL;
-    }
-}
+    for (i = 0; i < 8; i++)
+      {
+	if (childs[i])
+	  childs[i]->DeleteChilds();
+	delete childs[i];
+	childs[i] = NULL;
+      }
+  }
 
 
-BlockAllocator ADTreeNode3F :: ball(sizeof (ADTreeNode3F));
+  BlockAllocator ADTreeNode3F :: ball(sizeof (ADTreeNode3F));
 
-void * ADTreeNode3F :: operator new(size_t)
-{
-  return ball.Alloc();
-}
+  void * ADTreeNode3F :: operator new(size_t)
+  {
+    return ball.Alloc();
+  }
 
-void ADTreeNode3F :: operator delete (void * p)
-{
-  ball.Free (p);
-}
+  void ADTreeNode3F :: operator delete (void * p)
+  {
+    ball.Free (p);
+  }
 
 
 
@@ -1076,211 +1043,206 @@ void ADTreeNode3F :: operator delete (void * p)
 
 
 
-ADTree3F :: ADTree3F (const float * acmin, 
-		      const float * acmax)
-  : ela(0)
-{
-  memcpy (cmin, acmin, 3 * sizeof(float));
-  memcpy (cmax, acmax, 3 * sizeof(float));
+  ADTree3F :: ADTree3F (const float * acmin, 
+			const float * acmax)
+    : ela(0)
+  {
+    memcpy (cmin, acmin, 3 * sizeof(float));
+    memcpy (cmax, acmax, 3 * sizeof(float));
 
-  root = new ADTreeNode3F;
-  for (int i = 0; i < 3; i++)
-    root->sep[i] = (cmin[i] + cmax[i]) / 2;
-}
+    root = new ADTreeNode3F;
+    for (int i = 0; i < 3; i++)
+      root->sep[i] = (cmin[i] + cmax[i]) / 2;
+  }
 
-ADTree3F :: ~ADTree3F ()
-{
-  root->DeleteChilds();
-  delete root;
-}
+  ADTree3F :: ~ADTree3F ()
+  {
+    root->DeleteChilds();
+    delete root;
+  }
 
 
-void ADTree3F :: Insert (const float * p, int pi)
-{
-  ADTreeNode3F *node;
-  ADTreeNode3F *next;
-  int lr;
+  void ADTree3F :: Insert (const float * p, int pi)
+  {
+    ADTreeNode3F *node;
+    ADTreeNode3F *next;
+    int lr;
 
-  float bmin[3];
-  float bmax[3];
-  int i, dir;
+    float bmin[3];
+    float bmax[3];
+    int i, dir;
   
-  memcpy (bmin, cmin, 3 * sizeof(float));
-  memcpy (bmax, cmax, 3 * sizeof(float));
+    memcpy (bmin, cmin, 3 * sizeof(float));
+    memcpy (bmax, cmax, 3 * sizeof(float));
 
-#ifdef MARK
-  //  MARK (insertloop3)
-#endif
 
-  next = root;
-  while (next)
-    {
-      node = next;
+    next = root;
+    while (next)
+      {
+	node = next;
       
-      if (!node->pi)
-	{    
-	  memcpy (node->data, p, 3 * sizeof(float));
-	  node->pi = pi;
+	if (!node->pi)
+	  {    
+	    memcpy (node->data, p, 3 * sizeof(float));
+	    node->pi = pi;
 
-	  if (ela.Size() < pi)
-	    ela.SetSize (pi);
-	  ela.Elem(pi) = node;
+	    if (ela.Size() < pi)
+	      ela.SetSize (pi);
+	    ela.Elem(pi) = node;
 
-	  return;
-	}
+	    return;
+	  }
 
-      dir = 0;
-      for (i = 0; i < 3; i++)
-	{
-	  if (node->sep[i] > p[i])
-	    {
-	      bmax[i] = node->sep[i];
-	    }
-	  else
-	    {
-	      bmin[i] = node->sep[i];
-	      dir += (1 << i);
-	    }
-	}
-      next = node->childs[dir];
+	dir = 0;
+	for (i = 0; i < 3; i++)
+	  {
+	    if (node->sep[i] > p[i])
+	      {
+		bmax[i] = node->sep[i];
+	      }
+	    else
+	      {
+		bmin[i] = node->sep[i];
+		dir += (1 << i);
+	      }
+	  }
+	next = node->childs[dir];
 
-      /*
-      if (node->sep > p[dir])
-	{
+	/*
+	  if (node->sep > p[dir])
+	  {
 	  next = node->left;
 	  bmax[dir] = node->sep;
 	  lr = 0;
-	}
-      else
-	{
+	  }
+	  else
+	  {
 	  next = node->right;
 	  bmin[dir] = node->sep;
 	  lr = 1;
-	}
-      */
-    }
+	  }
+	*/
+      }
 
 
-  next = new ADTreeNode3F;
-  memcpy (next->data, p, 3 * sizeof(float));
-  next->pi = pi;
+    next = new ADTreeNode3F;
+    memcpy (next->data, p, 3 * sizeof(float));
+    next->pi = pi;
 
-  for (i = 0; i < 3; i++)
-    next->sep[i] = (bmin[i] + bmax[i]) / 2;
+    for (i = 0; i < 3; i++)
+      next->sep[i] = (bmin[i] + bmax[i]) / 2;
   
 
-  if (ela.Size() < pi)
-    ela.SetSize (pi);
-  ela.Elem(pi) = next;
+    if (ela.Size() < pi)
+      ela.SetSize (pi);
+    ela.Elem(pi) = next;
 
-  node->childs[dir] = next;
-  next->father = node;
+    node->childs[dir] = next;
+    next->father = node;
 
-  while (node)
-    {
-      node->nchilds++;
-      node = node->father;
-    }
-}
+    while (node)
+      {
+	node->nchilds++;
+	node = node->father;
+      }
+  }
 
-void ADTree3F :: DeleteElement (int pi)
-{
-  ADTreeNode3F * node = ela.Get(pi);
+  void ADTree3F :: DeleteElement (int pi)
+  {
+    ADTreeNode3F * node = ela.Get(pi);
 
-  node->pi = 0;
+    node->pi = 0;
 
-  node = node->father;
-  while (node)
-    {
-      node->nchilds--;
-      node = node->father;
-    }
-}
+    node = node->father;
+    while (node)
+      {
+	node->nchilds--;
+	node = node->father;
+      }
+  }
 
-void ADTree3F :: GetIntersecting (const float * bmin, 
-				  const float * bmax,
-				  ARRAY<int> & pis) const
-{
-  static ARRAY<ADTreeNode3F*> stack(1000);
-  ADTreeNode3F * node;
-  int dir, i, stacks;
+  void ADTree3F :: GetIntersecting (const float * bmin, 
+				    const float * bmax,
+				    ARRAY<int> & pis) const
+  {
+    static ARRAY<ADTreeNode3F*> stack(1000);
+    ADTreeNode3F * node;
+    int dir, i, stacks;
 
-  stack.SetSize (1000);
-  pis.SetSize(0);
+    stack.SetSize (1000);
+    pis.SetSize(0);
 
-  stack.Elem(1) = root;
-  stacks = 1;
+    stack.Elem(1) = root;
+    stacks = 1;
 
-  while (stacks)
-    {
-      node = stack.Get(stacks);
-      stacks--;
+    while (stacks)
+      {
+	node = stack.Get(stacks);
+	stacks--;
 
-      if (node->pi)
-	{
-	  if (node->data[0] >= bmin[0] && node->data[0] <= bmax[0] &&
-	      node->data[1] >= bmin[1] && node->data[1] <= bmax[1] &&
-	      node->data[2] >= bmin[2] && node->data[2] <= bmax[2])
+	if (node->pi)
+	  {
+	    if (node->data[0] >= bmin[0] && node->data[0] <= bmax[0] &&
+		node->data[1] >= bmin[1] && node->data[1] <= bmax[1] &&
+		node->data[2] >= bmin[2] && node->data[2] <= bmax[2])
 
-	    pis.Append (node->pi);
-	}
+	      pis.Append (node->pi);
+	  }
 
       
-      int i1min = (bmin[0] <= node->sep[0]) ? 0 : 1;
-      int i1max = (bmax[0] < node->sep[0]) ? 0 : 1;
-      int i2min = (bmin[1] <= node->sep[1]) ? 0 : 1;
-      int i2max = (bmax[1] < node->sep[1]) ? 0 : 1;
-      int i3min = (bmin[2] <= node->sep[2]) ? 0 : 1;
-      int i3max = (bmax[2] < node->sep[2]) ? 0 : 1;
-
-      int i1, i2, i3;
-      for (i1 = i1min; i1 <= i1max; i1++)
-	for (i2 = i2min; i2 <= i2max; i2++)
-	  for (i3 = i3min; i3 <= i3max; i3++)
-	    {
-	      i = i1+2*i2+4*i3;
-	      if (node->childs[i])
-		{
-		  stacks++;
-		  stack.Elem(stacks) = node->childs[i];
-		}
-	    }
+	int i1min = (bmin[0] <= node->sep[0]) ? 0 : 1;
+	int i1max = (bmax[0] < node->sep[0]) ? 0 : 1;
+	int i2min = (bmin[1] <= node->sep[1]) ? 0 : 1;
+	int i2max = (bmax[1] < node->sep[1]) ? 0 : 1;
+	int i3min = (bmin[2] <= node->sep[2]) ? 0 : 1;
+	int i3max = (bmax[2] < node->sep[2]) ? 0 : 1;
+
+	int i1, i2, i3;
+	for (i1 = i1min; i1 <= i1max; i1++)
+	  for (i2 = i2min; i2 <= i2max; i2++)
+	    for (i3 = i3min; i3 <= i3max; i3++)
+	      {
+		i = i1+2*i2+4*i3;
+		if (node->childs[i])
+		  {
+		    stacks++;
+		    stack.Elem(stacks) = node->childs[i];
+		  }
+	      }
       
-      /*
-      if (node->left && bmin[dir] <= node->sep)
-	{
+	/*
+	  if (node->left && bmin[dir] <= node->sep)
+	  {
 	  stacks++;
 	  stack.Elem(stacks) = node->left;
 	  stackdir.Elem(stacks) = ndir;
-	}
-      if (node->right && bmax[dir] >= node->sep)
-	{
+	  }
+	  if (node->right && bmax[dir] >= node->sep)
+	  {
 	  stacks++;
 	  stack.Elem(stacks) = node->right;
 	  stackdir.Elem(stacks) = ndir;
-	}
-      */
-    }
-}
-
-void ADTree3F :: PrintRec (ostream & ost, const ADTreeNode3F * node) const
-{
-  int i;
-  if (node->data)
-    {
-      ost << node->pi << ": ";
-      ost << node->nchilds << " childs, ";
-      for (i = 0; i < 3; i++)
-	ost << node->data[i] << " ";
-      ost << endl;
-    }
-
-  for (i = 0; i < 8; i++)
-    if (node->childs[i])
-      PrintRec (ost, node->childs[i]);
-}
+	  }
+	*/
+      }
+  }
 
+  void ADTree3F :: PrintRec (ostream & ost, const ADTreeNode3F * node) const
+  {
+    int i;
+    if (node->data)
+      {
+	ost << node->pi << ": ";
+	ost << node->nchilds << " childs, ";
+	for (i = 0; i < 3; i++)
+	  ost << node->data[i] << " ";
+	ost << endl;
+      }
 
+    for (i = 0; i < 8; i++)
+      if (node->childs[i])
+	PrintRec (ost, node->childs[i]);
+  }
 
 
 
@@ -1292,944 +1254,935 @@ void ADTree3F :: PrintRec (ostream & ost, const ADTreeNode3F * node) const
 
 
 
-/* ******************************* ADTree3FM ******************************* */
 
 
-ADTreeNode3FM :: ADTreeNode3FM()
-{
-  father = NULL;
-  nchilds = 0;
-  int i;
+  /* ******************************* ADTree3FM ******************************* */
 
-  for (i = 0; i < ADTN_SIZE; i++)
-    pi[i] = 0;
 
-  for (i = 0; i < 8; i++)
-    childs[i] = NULL;
-}
+  ADTreeNode3FM :: ADTreeNode3FM()
+  {
+    father = NULL;
+    nchilds = 0;
+    int i;
 
-void ADTreeNode3FM :: DeleteChilds ()
-{
-  int i;
+    for (i = 0; i < ADTN_SIZE; i++)
+      pi[i] = 0;
 
-  for (i = 0; i < 8; i++)
-    {
-      if (childs[i])
-	childs[i]->DeleteChilds();
-      delete childs[i];
+    for (i = 0; i < 8; i++)
       childs[i] = NULL;
-    }
-}
+  }
 
+  void ADTreeNode3FM :: DeleteChilds ()
+  {
+    int i;
 
-BlockAllocator ADTreeNode3FM :: ball(sizeof (ADTreeNode3FM));
-
-void * ADTreeNode3FM :: operator new(size_t)
-{
-  return ball.Alloc();
-}
-
-void ADTreeNode3FM :: operator delete (void * p)
-{
-  ball.Free (p);
-}
-
+    for (i = 0; i < 8; i++)
+      {
+	if (childs[i])
+	  childs[i]->DeleteChilds();
+	delete childs[i];
+	childs[i] = NULL;
+      }
+  }
 
 
+  BlockAllocator ADTreeNode3FM :: ball(sizeof (ADTreeNode3FM));
 
+  void * ADTreeNode3FM :: operator new(size_t)
+  {
+    return ball.Alloc();
+  }
 
+  void ADTreeNode3FM :: operator delete (void * p)
+  {
+    ball.Free (p);
+  }
 
 
-ADTree3FM :: ADTree3FM (const float * acmin, 
-		      const float * acmax)
-  : ela(0)
-{
-  memcpy (cmin, acmin, 3 * sizeof(float));
-  memcpy (cmax, acmax, 3 * sizeof(float));
 
-  root = new ADTreeNode3FM;
-  for (int i = 0; i < 3; i++)
-    root->sep[i] = (cmin[i] + cmax[i]) / 2;
-}
 
-ADTree3FM :: ~ADTree3FM ()
-{
-  root->DeleteChilds();
-  delete root;
-}
 
 
-void ADTree3FM :: Insert (const float * p, int pi)
-{
-  ADTreeNode3FM *node;
-  ADTreeNode3FM *next;
-  int lr;
 
-  float bmin[3];
-  float bmax[3];
-  int i, dir;
+  ADTree3FM :: ADTree3FM (const float * acmin, 
+			  const float * acmax)
+    : ela(0)
+  {
+    memcpy (cmin, acmin, 3 * sizeof(float));
+    memcpy (cmax, acmax, 3 * sizeof(float));
+
+    root = new ADTreeNode3FM;
+    for (int i = 0; i < 3; i++)
+      root->sep[i] = (cmin[i] + cmax[i]) / 2;
+  }
+
+  ADTree3FM :: ~ADTree3FM ()
+  {
+    root->DeleteChilds();
+    delete root;
+  }
+
+
+  void ADTree3FM :: Insert (const float * p, int pi)
+  {
+    ADTreeNode3FM *node;
+    ADTreeNode3FM *next;
+    int lr;
+
+    float bmin[3];
+    float bmax[3];
+    int i, dir;
   
-  memcpy (bmin, cmin, 3 * sizeof(float));
-  memcpy (bmax, cmax, 3 * sizeof(float));
-
-#ifdef MARK
-  //  MARK (insertloop3)
-#endif
+    memcpy (bmin, cmin, 3 * sizeof(float));
+    memcpy (bmax, cmax, 3 * sizeof(float));
 
-  next = root;
-  while (next)
-    {
-      node = next;
+    next = root;
+    while (next)
+      {
+	node = next;
       
-      for (i = 0; i < ADTN_SIZE; i++)
-	if (!node->pi[i])
-	  {    
-	    memcpy (node->data[i], p, 3 * sizeof(float));
-	    node->pi[i] = pi;
+	for (i = 0; i < ADTN_SIZE; i++)
+	  if (!node->pi[i])
+	    {    
+	      memcpy (node->data[i], p, 3 * sizeof(float));
+	      node->pi[i] = pi;
 	    
-	    if (ela.Size() < pi)
-	      ela.SetSize (pi);
-	    ela.Elem(pi) = node;
+	      if (ela.Size() < pi)
+		ela.SetSize (pi);
+	      ela.Elem(pi) = node;
 	    
-	    return;
-	  }
-
-      dir = 0;
-      for (i = 0; i < 3; i++)
-	{
-	  if (node->sep[i] > p[i])
-	    {
-	      bmax[i] = node->sep[i];
-	    }
-	  else
-	    {
-	      bmin[i] = node->sep[i];
-	      dir += (1 << i);
+	      return;
 	    }
-	}
-      next = node->childs[dir];
 
-      /*
-      if (node->sep > p[dir])
-	{
+	dir = 0;
+	for (i = 0; i < 3; i++)
+	  {
+	    if (node->sep[i] > p[i])
+	      {
+		bmax[i] = node->sep[i];
+	      }
+	    else
+	      {
+		bmin[i] = node->sep[i];
+		dir += (1 << i);
+	      }
+	  }
+	next = node->childs[dir];
+
+	/*
+	  if (node->sep > p[dir])
+	  {
 	  next = node->left;
 	  bmax[dir] = node->sep;
 	  lr = 0;
-	}
-      else
-	{
+	  }
+	  else
+	  {
 	  next = node->right;
 	  bmin[dir] = node->sep;
 	  lr = 1;
-	}
-      */
-    }
+	  }
+	*/
+      }
 
 
-  next = new ADTreeNode3FM;
-  memcpy (next->data[0], p, 3 * sizeof(float));
-  next->pi[0] = pi;
+    next = new ADTreeNode3FM;
+    memcpy (next->data[0], p, 3 * sizeof(float));
+    next->pi[0] = pi;
 
-  for (i = 0; i < 3; i++)
-    next->sep[i] = (bmin[i] + bmax[i]) / 2;
+    for (i = 0; i < 3; i++)
+      next->sep[i] = (bmin[i] + bmax[i]) / 2;
   
 
-  if (ela.Size() < pi)
-    ela.SetSize (pi);
-  ela.Elem(pi) = next;
+    if (ela.Size() < pi)
+      ela.SetSize (pi);
+    ela.Elem(pi) = next;
 
-  node->childs[dir] = next;
-  next->father = node;
+    node->childs[dir] = next;
+    next->father = node;
 
-  while (node)
-    {
-      node->nchilds++;
-      node = node->father;
-    }
-}
+    while (node)
+      {
+	node->nchilds++;
+	node = node->father;
+      }
+  }
 
-void ADTree3FM :: DeleteElement (int pi)
-{
-  ADTreeNode3FM * node = ela.Get(pi);
+  void ADTree3FM :: DeleteElement (int pi)
+  {
+    ADTreeNode3FM * node = ela.Get(pi);
 
-  int i;
-  for (i = 0; i < ADTN_SIZE; i++)
-    if (node->pi[i] == pi)
-      node->pi[i] = 0;
+    int i;
+    for (i = 0; i < ADTN_SIZE; i++)
+      if (node->pi[i] == pi)
+	node->pi[i] = 0;
 
-  node = node->father;
-  while (node)
-    {
-      node->nchilds--;
-      node = node->father;
-    }
-}
+    node = node->father;
+    while (node)
+      {
+	node->nchilds--;
+	node = node->father;
+      }
+  }
 
-void ADTree3FM :: GetIntersecting (const float * bmin, 
-				  const float * bmax,
-				  ARRAY<int> & pis) const
-{
-  static ARRAY<ADTreeNode3FM*> stack(1000);
-  ADTreeNode3FM * node;
-  int dir, i, stacks;
+  void ADTree3FM :: GetIntersecting (const float * bmin, 
+				     const float * bmax,
+				     ARRAY<int> & pis) const
+  {
+    static ARRAY<ADTreeNode3FM*> stack(1000);
+    ADTreeNode3FM * node;
+    int dir, i, stacks;
 
-  stack.SetSize (1000);
-  pis.SetSize(0);
+    stack.SetSize (1000);
+    pis.SetSize(0);
 
-  stack.Elem(1) = root;
-  stacks = 1;
+    stack.Elem(1) = root;
+    stacks = 1;
 
-  while (stacks)
-    {
-      node = stack.Get(stacks);
-      stacks--;
+    while (stacks)
+      {
+	node = stack.Get(stacks);
+	stacks--;
 
-      int * hpi = node->pi;
-      for (i = 0; i < ADTN_SIZE; i++)
-	if (hpi[i])
-	  {
-	    float * datai = &node->data[i][0];
-	    if (datai[0] >= bmin[0] && datai[0] <= bmax[0] &&
-		datai[1] >= bmin[1] && datai[1] <= bmax[1] &&
-		datai[2] >= bmin[2] && datai[2] <= bmax[2])
+	int * hpi = node->pi;
+	for (i = 0; i < ADTN_SIZE; i++)
+	  if (hpi[i])
+	    {
+	      float * datai = &node->data[i][0];
+	      if (datai[0] >= bmin[0] && datai[0] <= bmax[0] &&
+		  datai[1] >= bmin[1] && datai[1] <= bmax[1] &&
+		  datai[2] >= bmin[2] && datai[2] <= bmax[2])
 	      
-	      pis.Append (node->pi[i]);
-	  }
+		pis.Append (node->pi[i]);
+	    }
 
-      /*
-      if (node->pi)
-	{
+	/*
+	  if (node->pi)
+	  {
 	  if (node->data[0] >= bmin[0] && node->data[0] <= bmax[0] &&
-	      node->data[1] >= bmin[1] && node->data[1] <= bmax[1] &&
-	      node->data[2] >= bmin[2] && node->data[2] <= bmax[2])
+	  node->data[1] >= bmin[1] && node->data[1] <= bmax[1] &&
+	  node->data[2] >= bmin[2] && node->data[2] <= bmax[2])
 
-	    pis.Append (node->pi);
-	}
-      */
+	  pis.Append (node->pi);
+	  }
+	*/
       
-      int i1min = (bmin[0] <= node->sep[0]) ? 0 : 1;
-      int i1max = (bmax[0] < node->sep[0]) ? 0 : 1;
-      int i2min = (bmin[1] <= node->sep[1]) ? 0 : 1;
-      int i2max = (bmax[1] < node->sep[1]) ? 0 : 1;
-      int i3min = (bmin[2] <= node->sep[2]) ? 0 : 1;
-      int i3max = (bmax[2] < node->sep[2]) ? 0 : 1;
-
-      int i1, i2, i3;
-      for (i1 = i1min; i1 <= i1max; i1++)
-	for (i2 = i2min; i2 <= i2max; i2++)
-	  for (i3 = i3min; i3 <= i3max; i3++)
-	    {
-	      i = i1+2*i2+4*i3;
-	      if (node->childs[i])
-		{
-		  stacks++;
-		  stack.Elem(stacks) = node->childs[i];
-		}
-	    }
+	int i1min = (bmin[0] <= node->sep[0]) ? 0 : 1;
+	int i1max = (bmax[0] < node->sep[0]) ? 0 : 1;
+	int i2min = (bmin[1] <= node->sep[1]) ? 0 : 1;
+	int i2max = (bmax[1] < node->sep[1]) ? 0 : 1;
+	int i3min = (bmin[2] <= node->sep[2]) ? 0 : 1;
+	int i3max = (bmax[2] < node->sep[2]) ? 0 : 1;
+
+	int i1, i2, i3;
+	for (i1 = i1min; i1 <= i1max; i1++)
+	  for (i2 = i2min; i2 <= i2max; i2++)
+	    for (i3 = i3min; i3 <= i3max; i3++)
+	      {
+		i = i1+2*i2+4*i3;
+		if (node->childs[i])
+		  {
+		    stacks++;
+		    stack.Elem(stacks) = node->childs[i];
+		  }
+	      }
       
-      /*
-      if (node->left && bmin[dir] <= node->sep)
-	{
+	/*
+	  if (node->left && bmin[dir] <= node->sep)
+	  {
 	  stacks++;
 	  stack.Elem(stacks) = node->left;
 	  stackdir.Elem(stacks) = ndir;
-	}
-      if (node->right && bmax[dir] >= node->sep)
-	{
+	  }
+	  if (node->right && bmax[dir] >= node->sep)
+	  {
 	  stacks++;
 	  stack.Elem(stacks) = node->right;
 	  stackdir.Elem(stacks) = ndir;
-	}
-      */
-    }
-}
-
-void ADTree3FM :: PrintRec (ostream & ost, const ADTreeNode3FM * node) const
-{
-  int i;
-  if (node->data)
-    {
-      ost << node->pi << ": ";
-      ost << node->nchilds << " childs, ";
-      for (i = 0; i < 3; i++)
-	ost << node->data[i] << " ";
-      ost << endl;
-    }
-
-  for (i = 0; i < 8; i++)
-    if (node->childs[i])
-      PrintRec (ost, node->childs[i]);
-}
-
-
-
-
+	  }
+	*/
+      }
+  }
 
+  void ADTree3FM :: PrintRec (ostream & ost, const ADTreeNode3FM * node) const
+  {
+    int i;
+    if (node->data)
+      {
+	ost << node->pi << ": ";
+	ost << node->nchilds << " childs, ";
+	for (i = 0; i < 3; i++)
+	  ost << node->data[i] << " ";
+	ost << endl;
+      }
 
+    for (i = 0; i < 8; i++)
+      if (node->childs[i])
+	PrintRec (ost, node->childs[i]);
+  }
 
 
 
 
-/* ******************************* ADTree6 ******************************* */
+#endif
 
 
-ADTreeNode6 :: ADTreeNode6()
-{
-  pi = 0;
 
-  left = NULL;
-  right = NULL;
-  father = NULL;
-  nchilds = 0;
-}
 
-void ADTreeNode6 :: DeleteChilds ()
-{
-  if (left)
-    {
-      left->DeleteChilds();
-      delete left;
-      left = NULL;
-    }
-  if (right)
-    {
-      right->DeleteChilds();
-      delete right;
-      right = NULL;
-    }
-}
 
 
-BlockAllocator ADTreeNode6 :: ball (sizeof (ADTreeNode6));
-void * ADTreeNode6 :: operator new(size_t)
-{
-  return ball.Alloc();
-}
+  /* ******************************* ADTree6 ******************************* */
 
-void ADTreeNode6 :: operator delete (void * p)
-{
-  ball.Free (p);
-}
 
+  ADTreeNode6 :: ADTreeNode6()
+  {
+    pi = -1;
 
+    left = NULL;
+    right = NULL;
+    father = NULL;
+    nchilds = 0;
+  }
 
+  void ADTreeNode6 :: DeleteChilds ()
+  {
+    if (left)
+      {
+	left->DeleteChilds();
+	delete left;
+	left = NULL;
+      }
+    if (right)
+      {
+	right->DeleteChilds();
+	delete right;
+	right = NULL;
+      }
+  }
 
 
-ADTree6 :: ADTree6 (const float * acmin, 
-			 const float * acmax)
-  : ela(0)
-{
-  memcpy (cmin, acmin, 6 * sizeof(float));
-  memcpy (cmax, acmax, 6 * sizeof(float));
+  BlockAllocator ADTreeNode6 :: ball (sizeof (ADTreeNode6));
+  void * ADTreeNode6 :: operator new(size_t)
+  {
+    return ball.Alloc();
+  }
 
-  root = new ADTreeNode6;
-  root->sep = (cmin[0] + cmax[0]) / 2;
-}
+  void ADTreeNode6 :: operator delete (void * p)
+  {
+    ball.Free (p);
+  }
 
-ADTree6 :: ~ADTree6 ()
-{
-  root->DeleteChilds();
-  delete root;
-}
 
-void ADTree6 :: Insert (const float * p, int pi)
-{
-  ADTreeNode6 *node;
-  ADTreeNode6 *next;
-  int dir;
-  int lr;
 
-  float bmin[6];
-  float bmax[6];
 
-  
-  memcpy (bmin, cmin, 6 * sizeof(float));
-  memcpy (bmax, cmax, 6 * sizeof(float));
 
-#ifdef MARK
-  MARK (insertloop6)
-#endif
+  ADTree6 :: ADTree6 (const float * acmin, 
+		      const float * acmax)
+    : ela(0)
+  {
+    memcpy (cmin, acmin, 6 * sizeof(float));
+    memcpy (cmax, acmax, 6 * sizeof(float));
+
+    root = new ADTreeNode6;
+    root->sep = (cmin[0] + cmax[0]) / 2;
+  }
+
+  ADTree6 :: ~ADTree6 ()
+  {
+    root->DeleteChilds();
+    delete root;
+  }
+
+  void ADTree6 :: Insert (const float * p, int pi)
+  {
+    ADTreeNode6 *node;
+    ADTreeNode6 *next;
+    int dir;
+    int lr;
+
+    float bmin[6];
+    float bmax[6];
 
-  next = root;
-  dir = 0;
-  while (next)
-    {
-      node = next;
+  
+    memcpy (bmin, cmin, 6 * sizeof(float));
+    memcpy (bmax, cmax, 6 * sizeof(float));
 
-      if (!node->pi)
-	{    
-	  memcpy (node->data, p, 6 * sizeof(float));
-	  node->pi = pi;
+    next = root;
+    dir = 0;
+    while (next)
+      {
+	node = next;
 
-	  if (ela.Size() < pi)
-	    ela.SetSize (pi);
-	  ela.Elem(pi) = node;
+	if (node->pi == -1)
+	  {    
+	    memcpy (node->data, p, 6 * sizeof(float));
+	    node->pi = pi;
 
-	  return;
-	}
+	    if (ela.Size() < pi+1)
+	      ela.SetSize (pi+1);
+	    ela[pi] = node;
 
-      if (node->sep > p[dir])
-	{
-	  next = node->left;
-	  bmax[dir] = node->sep;
-	  lr = 0;
-	}
-      else
-	{
-	  next = node->right;
-	  bmin[dir] = node->sep;
-	  lr = 1;
-	}
+	    return;
+	  }
 
-      dir++;
-      if (dir == 6)
-	dir = 0;
-    }
+	if (node->sep > p[dir])
+	  {
+	    next = node->left;
+	    bmax[dir] = node->sep;
+	    lr = 0;
+	  }
+	else
+	  {
+	    next = node->right;
+	    bmin[dir] = node->sep;
+	    lr = 1;
+	  }
 
-#ifdef MARK
-  MARK (insertend6)
-#endif
+	dir++;
+	if (dir == 6)
+	  dir = 0;
+      }
 
 
-  next = new ADTreeNode6;
-  memcpy (next->data, p, 6 * sizeof(float));
-  next->pi = pi;
-  next->sep = (bmin[dir] + bmax[dir]) / 2;
+    next = new ADTreeNode6;
+    memcpy (next->data, p, 6 * sizeof(float));
+    next->pi = pi;
+    next->sep = (bmin[dir] + bmax[dir]) / 2;
 
 
-  if (ela.Size() < pi)
-    ela.SetSize (pi);
-  ela.Elem(pi) = next;
+    if (ela.Size() < pi+1)
+      ela.SetSize (pi+1);
+    ela[pi] = next;
 
 
-  if (lr)
-    node->right = next;
-  else
-    node->left = next;
-  next -> father = node;
+    if (lr)
+      node->right = next;
+    else
+      node->left = next;
+    next -> father = node;
 
-  while (node)
-    {
-      node->nchilds++;
-      node = node->father;
-    }
-}
+    while (node)
+      {
+	node->nchilds++;
+	node = node->father;
+      }
+  }
 
-void ADTree6 :: DeleteElement (int pi)
-{
-  ADTreeNode6 * node = ela.Get(pi);
+  void ADTree6 :: DeleteElement (int pi)
+  {
+    ADTreeNode6 * node = ela[pi];
 
-  node->pi = 0;
+    node->pi = -1;
 
-  node = node->father;
-  while (node)
-    {
-      node->nchilds--;
-      node = node->father;
-    }
-}
+    node = node->father;
+    while (node)
+      {
+	node->nchilds--;
+	node = node->father;
+      }
+  }
 
-void ADTree6 :: PrintMemInfo (ostream & ost) const
-{
-  ost << Elements() << " elements a " << sizeof(ADTreeNode6) 
-      << " Bytes = "
-      << Elements() * sizeof(ADTreeNode6) << endl;
-  ost << "maxind = " << ela.Size() << " = " << sizeof(ADTreeNode6*) * ela.Size() << " Bytes" << endl;
-}
+  void ADTree6 :: PrintMemInfo (ostream & ost) const
+  {
+    ost << Elements() << " elements a " << sizeof(ADTreeNode6) 
+	<< " Bytes = "
+	<< Elements() * sizeof(ADTreeNode6) << endl;
+    ost << "maxind = " << ela.Size() << " = " << sizeof(ADTreeNode6*) * ela.Size() << " Bytes" << endl;
+  }
 
 
 
-class inttn6 {
-public:
-  int dir;
-  ADTreeNode6 * node;
-};
+  class inttn6 {
+  public:
+    int dir;
+    ADTreeNode6 * node;
+  };
 
 
 
 
-void ADTree6 :: GetIntersecting (const float * bmin, 
-				 const float * bmax,
-				 ARRAY<int> & pis) const
-{
-  static ARRAY<inttn6> stack(10000);
-  ADTreeNode6 * node;
-  int dir, stacks;
+  void ADTree6 :: GetIntersecting (const float * bmin, 
+				   const float * bmax,
+				   ARRAY<int> & pis) const
+  {
+    static ARRAY<inttn6> stack(10000);
+    ADTreeNode6 * node;
+    int dir, stacks;
 
-  stack.SetSize (10000);
-  pis.SetSize(0);
+    stack.SetSize (10000);
+    pis.SetSize(0);
 
-  stack.Elem(1).node = root;
-  stack.Elem(1).dir = 0;
-  stacks = 1;
+    stack.Elem(1).node = root;
+    stack.Elem(1).dir = 0;
+    stacks = 1;
 
-  while (stacks)
-    {
-      node = stack.Get(stacks).node;
-      dir = stack.Get(stacks).dir; 
-      stacks--;
+    while (stacks)
+      {
+	node = stack.Get(stacks).node;
+	dir = stack.Get(stacks).dir; 
+	stacks--;
 
-      if (node->pi)
-	{
+	if (node->pi != -1)
+	  {
 	  
-// 	  int in = 1;
-// 	  for (i = 0; i < 3; i++)
-// 	    if (/* node->data[i] < bmin[i]  || */ node->data[i] > bmax[i] || 
-// 		node->data[i+3] < bmin[i+3] /* || node->data[i+3] > bmax[i+3] */ )
-// 	      {
-// 		in = 0;
-// 		break;
-// 	      }
-
-// 	  if (in)
-// 	    pis.Append (node->pi);
-
-	  if (node->data[0] > bmax[0] || 
-	      node->data[1] > bmax[1] || 
-	      node->data[2] > bmax[2] || 
-	      node->data[3] < bmin[3] || 
-	      node->data[4] < bmin[4] || 
-	      node->data[5] < bmin[5])
-	    ;
-	  else
-	    pis.Append (node->pi);
-	}
+	    // 	  int in = 1;
+	    // 	  for (i = 0; i < 3; i++)
+	    // 	    if (/* node->data[i] < bmin[i]  || */ node->data[i] > bmax[i] || 
+	    // 		node->data[i+3] < bmin[i+3] /* || node->data[i+3] > bmax[i+3] */ )
+	    // 	      {
+	    // 		in = 0;
+	    // 		break;
+	    // 	      }
+
+	    // 	  if (in)
+	    // 	    pis.Append (node->pi);
+
+	    if (node->data[0] > bmax[0] || 
+		node->data[1] > bmax[1] || 
+		node->data[2] > bmax[2] || 
+		node->data[3] < bmin[3] || 
+		node->data[4] < bmin[4] || 
+		node->data[5] < bmin[5])
+	      ;
+	    else
+	      pis.Append (node->pi);
+	  }
 
 
-      int ndir = dir+1;
-      if (ndir == 6)
-	ndir = 0;
+	int ndir = dir+1;
+	if (ndir == 6)
+	  ndir = 0;
 
-      if (node->left && bmin[dir] <= node->sep)
-	{
-	  stacks++;
-	  stack.Elem(stacks).node = node->left;
-	  stack.Elem(stacks).dir = ndir;
-	}
-      if (node->right && bmax[dir] >= node->sep)
-	{
-	  stacks++;
-	  stack.Elem(stacks).node = node->right;
-	  stack.Elem(stacks).dir = ndir;
-	}
-    }
-}
+	if (node->left && bmin[dir] <= node->sep)
+	  {
+	    stacks++;
+	    stack.Elem(stacks).node = node->left;
+	    stack.Elem(stacks).dir = ndir;
+	  }
+	if (node->right && bmax[dir] >= node->sep)
+	  {
+	    stacks++;
+	    stack.Elem(stacks).node = node->right;
+	    stack.Elem(stacks).dir = ndir;
+	  }
+      }
+  }
 
-/*
-void ADTree6 :: GetIntersecting (const float * bmin, 
-				 const float * bmax,
-				 ARRAY<int> & pis) const
-{
-  static ARRAY<ADTreeNode6*> stack(10000);
-  static ARRAY<int> stackdir(10000);
-  ADTreeNode6 * node;
-  int dir, stacks;
+  /*
+    void ADTree6 :: GetIntersecting (const float * bmin, 
+    const float * bmax,
+    ARRAY<int> & pis) const
+    {
+    static ARRAY<ADTreeNode6*> stack(10000);
+    static ARRAY<int> stackdir(10000);
+    ADTreeNode6 * node;
+    int dir, stacks;
 
-  stack.SetSize (10000);
-  stackdir.SetSize(10000);
-  pis.SetSize(0);
+    stack.SetSize (10000);
+    stackdir.SetSize(10000);
+    pis.SetSize(0);
 
-  stack.Elem(1) = root;
-  stackdir.Elem(1) = 0;
-  stacks = 1;
+    stack.Elem(1) = root;
+    stackdir.Elem(1) = 0;
+    stacks = 1;
 
-  while (stacks)
+    while (stacks)
     {
-      node = stack.Get(stacks);
-      dir = stackdir.Get(stacks); 
-      stacks--;
+    node = stack.Get(stacks);
+    dir = stackdir.Get(stacks); 
+    stacks--;
 
-      if (node->pi)
-	{
+    if (node->pi)
+    {
 	  
-	  if (node->data[0] > bmax[0] || 
-	      node->data[1] > bmax[1] || 
-	      node->data[2] > bmax[2] || 
-	      node->data[3] < bmin[3] || 
-	      node->data[4] < bmin[4] || 
-	      node->data[5] < bmin[5])
-	    ;
-	  else
-	    pis.Append (node->pi);
-	}
+    if (node->data[0] > bmax[0] || 
+    node->data[1] > bmax[1] || 
+    node->data[2] > bmax[2] || 
+    node->data[3] < bmin[3] || 
+    node->data[4] < bmin[4] || 
+    node->data[5] < bmin[5])
+    ;
+    else
+    pis.Append (node->pi);
+    }
 
 
-      int ndir = dir+1;
-      if (ndir == 6)
-	ndir = 0;
+    int ndir = dir+1;
+    if (ndir == 6)
+    ndir = 0;
 
-      if (node->left && bmin[dir] <= node->sep)
-	{
-	  stacks++;
-	  stack.Elem(stacks) = node->left;
-	  stackdir.Elem(stacks) = ndir;
-	}
-      if (node->right && bmax[dir] >= node->sep)
-	{
-	  stacks++;
-	  stack.Elem(stacks) = node->right;
-	  stackdir.Elem(stacks) = ndir;
-	}
+    if (node->left && bmin[dir] <= node->sep)
+    {
+    stacks++;
+    stack.Elem(stacks) = node->left;
+    stackdir.Elem(stacks) = ndir;
     }
-}
-*/
-
-
-void ADTree6 :: PrintRec (ostream & ost, const ADTreeNode6 * node) const
-{
-  
-  if (node->data)
+    if (node->right && bmax[dir] >= node->sep)
     {
-      ost << node->pi << ": ";
-      ost << node->nchilds << " childs, ";
-      for (int i = 0; i < 6; i++)
-	ost << node->data[i] << " ";
-      ost << endl;
+    stacks++;
+    stack.Elem(stacks) = node->right;
+    stackdir.Elem(stacks) = ndir;
     }
-  if (node->left)
-    PrintRec (ost, node->left);
-  if (node->right)
-    PrintRec (ost, node->right);
-}
+    }
+    }
+  */
 
 
-int ADTree6 :: DepthRec (const ADTreeNode6 * node) const
-{
-  int ldepth = 0;
-  int rdepth = 0;
-
-  if (node->left)
-    ldepth = DepthRec(node->left);
-  if (node->right)
-    rdepth = DepthRec(node->right);
-  return 1 + max2 (ldepth, rdepth);
-}
+  void ADTree6 :: PrintRec (ostream & ost, const ADTreeNode6 * node) const
+  {
+  
+    if (node->data)
+      {
+	ost << node->pi << ": ";
+	ost << node->nchilds << " childs, ";
+	for (int i = 0; i < 6; i++)
+	  ost << node->data[i] << " ";
+	ost << endl;
+      }
+    if (node->left)
+      PrintRec (ost, node->left);
+    if (node->right)
+      PrintRec (ost, node->right);
+  }
 
-int ADTree6 :: ElementsRec (const ADTreeNode6 * node) const
-{
-  int els = 1;
-  if (node->left)
-    els += ElementsRec(node->left);
-  if (node->right)
-    els += ElementsRec(node->right);
-  return els;
-}
 
+  int ADTree6 :: DepthRec (const ADTreeNode6 * node) const
+  {
+    int ldepth = 0;
+    int rdepth = 0;
 
+    if (node->left)
+      ldepth = DepthRec(node->left);
+    if (node->right)
+      rdepth = DepthRec(node->right);
+    return 1 + max2 (ldepth, rdepth);
+  }
 
+  int ADTree6 :: ElementsRec (const ADTreeNode6 * node) const
+  {
+    int els = 1;
+    if (node->left)
+      els += ElementsRec(node->left);
+    if (node->right)
+      els += ElementsRec(node->right);
+    return els;
+  }
 
 
 
 
-/* ******************************* ADTree6F ******************************* */
 
 
-ADTreeNode6F :: ADTreeNode6F()
-{
-  pi = 0;
-  father = NULL;
-  nchilds = 0;
-  int i;
-  for (i = 0; i < 64; i++)
-    childs[i] = NULL;
-}
+#ifdef ABC
 
-void ADTreeNode6F :: DeleteChilds ()
-{
-  int i;
+  /* ******************************* ADTree6F ******************************* */
 
-  for (i = 0; i < 64; i++)
-    {
-      if (childs[i])
-	childs[i]->DeleteChilds();
-      delete childs[i];
+
+  ADTreeNode6F :: ADTreeNode6F()
+  {
+    pi = 0;
+    father = NULL;
+    nchilds = 0;
+    int i;
+    for (i = 0; i < 64; i++)
       childs[i] = NULL;
-    }
-}
+  }
+
+  void ADTreeNode6F :: DeleteChilds ()
+  {
+    int i;
 
+    for (i = 0; i < 64; i++)
+      {
+	if (childs[i])
+	  childs[i]->DeleteChilds();
+	delete childs[i];
+	childs[i] = NULL;
+      }
+  }
 
-BlockAllocator ADTreeNode6F :: ball(sizeof (ADTreeNode6F));
 
-void * ADTreeNode6F :: operator new(size_t)
-{
-  return ball.Alloc();
-}
+  BlockAllocator ADTreeNode6F :: ball(sizeof (ADTreeNode6F));
 
-void ADTreeNode6F :: operator delete (void * p)
-{
-  ball.Free (p);
-}
+  void * ADTreeNode6F :: operator new(size_t)
+  {
+    return ball.Alloc();
+  }
 
+  void ADTreeNode6F :: operator delete (void * p)
+  {
+    ball.Free (p);
+  }
 
 
 
 
 
 
-ADTree6F :: ADTree6F (const float * acmin, 
-		      const float * acmax)
-  : ela(0)
-{
-  memcpy (cmin, acmin, 6 * sizeof(float));
-  memcpy (cmax, acmax, 6 * sizeof(float));
 
-  root = new ADTreeNode6F;
-  for (int i = 0; i < 6; i++)
-    root->sep[i] = (cmin[i] + cmax[i]) / 2;
-}
+  ADTree6F :: ADTree6F (const float * acmin, 
+			const float * acmax)
+    : ela(0)
+  {
+    memcpy (cmin, acmin, 6 * sizeof(float));
+    memcpy (cmax, acmax, 6 * sizeof(float));
 
-ADTree6F :: ~ADTree6F ()
-{
-  root->DeleteChilds();
-  delete root;
-}
+    root = new ADTreeNode6F;
+    for (int i = 0; i < 6; i++)
+      root->sep[i] = (cmin[i] + cmax[i]) / 2;
+  }
 
+  ADTree6F :: ~ADTree6F ()
+  {
+    root->DeleteChilds();
+    delete root;
+  }
 
-void ADTree6F :: Insert (const float * p, int pi)
-{
-  ADTreeNode6F *node;
-  ADTreeNode6F *next;
-  int lr;
 
-  float bmin[6];
-  float bmax[6];
-  int i, dir;
-  
-  memcpy (bmin, cmin, 6 * sizeof(float));
-  memcpy (bmax, cmax, 6 * sizeof(float));
+  void ADTree6F :: Insert (const float * p, int pi)
+  {
+    ADTreeNode6F *node;
+    ADTreeNode6F *next;
+    int lr;
 
-#ifdef MARK
-  //  MARK (insertloop3)
-#endif
+    float bmin[6];
+    float bmax[6];
+    int i, dir;
+  
+    memcpy (bmin, cmin, 6 * sizeof(float));
+    memcpy (bmax, cmax, 6 * sizeof(float));
 
-  next = root;
-  while (next)
-    {
-      node = next;
+    next = root;
+    while (next)
+      {
+	node = next;
       
-      if (!node->pi)
-	{    
-	  memcpy (node->data, p, 6 * sizeof(float));
-	  node->pi = pi;
+	if (!node->pi)
+	  {    
+	    memcpy (node->data, p, 6 * sizeof(float));
+	    node->pi = pi;
 
-	  if (ela.Size() < pi)
-	    ela.SetSize (pi);
-	  ela.Elem(pi) = node;
+	    if (ela.Size() < pi)
+	      ela.SetSize (pi);
+	    ela.Elem(pi) = node;
 
-	  return;
-	}
+	    return;
+	  }
 
-      dir = 0;
-      for (i = 0; i < 6; i++)
-	{
-	  if (node->sep[i] > p[i])
-	    {
-	      bmax[i] = node->sep[i];
-	    }
-	  else
-	    {
-	      bmin[i] = node->sep[i];
-	      dir += (1 << i);
-	    }
-	}
-      next = node->childs[dir];
+	dir = 0;
+	for (i = 0; i < 6; i++)
+	  {
+	    if (node->sep[i] > p[i])
+	      {
+		bmax[i] = node->sep[i];
+	      }
+	    else
+	      {
+		bmin[i] = node->sep[i];
+		dir += (1 << i);
+	      }
+	  }
+	next = node->childs[dir];
 
-      /*
-      if (node->sep > p[dir])
-	{
+	/*
+	  if (node->sep > p[dir])
+	  {
 	  next = node->left;
 	  bmax[dir] = node->sep;
 	  lr = 0;
-	}
-      else
-	{
+	  }
+	  else
+	  {
 	  next = node->right;
 	  bmin[dir] = node->sep;
 	  lr = 1;
-	}
-      */
-    }
+	  }
+	*/
+      }
 
 
-  next = new ADTreeNode6F;
-  memcpy (next->data, p, 6 * sizeof(float));
-  next->pi = pi;
+    next = new ADTreeNode6F;
+    memcpy (next->data, p, 6 * sizeof(float));
+    next->pi = pi;
 
-  for (i = 0; i < 6; i++)
-    next->sep[i] = (bmin[i] + bmax[i]) / 2;
+    for (i = 0; i < 6; i++)
+      next->sep[i] = (bmin[i] + bmax[i]) / 2;
   
 
-  if (ela.Size() < pi)
-    ela.SetSize (pi);
-  ela.Elem(pi) = next;
+    if (ela.Size() < pi)
+      ela.SetSize (pi);
+    ela.Elem(pi) = next;
 
-  node->childs[dir] = next;
-  next->father = node;
+    node->childs[dir] = next;
+    next->father = node;
 
-  while (node)
-    {
-      node->nchilds++;
-      node = node->father;
-    }
-}
+    while (node)
+      {
+	node->nchilds++;
+	node = node->father;
+      }
+  }
 
-void ADTree6F :: DeleteElement (int pi)
-{
-  ADTreeNode6F * node = ela.Get(pi);
+  void ADTree6F :: DeleteElement (int pi)
+  {
+    ADTreeNode6F * node = ela.Get(pi);
 
-  node->pi = 0;
+    node->pi = 0;
 
-  node = node->father;
-  while (node)
-    {
-      node->nchilds--;
-      node = node->father;
-    }
-}
+    node = node->father;
+    while (node)
+      {
+	node->nchilds--;
+	node = node->father;
+      }
+  }
 
-void ADTree6F :: GetIntersecting (const float * bmin, 
-				  const float * bmax,
-				  ARRAY<int> & pis) const
-{
-  static ARRAY<ADTreeNode6F*> stack(1000);
-  ADTreeNode6F * node;
-  int dir, i, stacks;
+  void ADTree6F :: GetIntersecting (const float * bmin, 
+				    const float * bmax,
+				    ARRAY<int> & pis) const
+  {
+    static ARRAY<ADTreeNode6F*> stack(1000);
+    ADTreeNode6F * node;
+    int dir, i, stacks;
 
-  stack.SetSize (1000);
-  pis.SetSize(0);
+    stack.SetSize (1000);
+    pis.SetSize(0);
 
-  stack.Elem(1) = root;
-  stacks = 1;
+    stack.Elem(1) = root;
+    stacks = 1;
 
-  while (stacks)
-    {
-      node = stack.Get(stacks);
-      stacks--;
+    while (stacks)
+      {
+	node = stack.Get(stacks);
+	stacks--;
 
-      if (node->pi)
-	{
-	  if (
-	      node->data[0] >= bmin[0] && node->data[0] <= bmax[0] &&
-	      node->data[1] >= bmin[1] && node->data[1] <= bmax[1] &&
-	      node->data[2] >= bmin[2] && node->data[2] <= bmax[2] &&
-	      node->data[3] >= bmin[3] && node->data[3] <= bmax[3] &&
-	      node->data[4] >= bmin[4] && node->data[4] <= bmax[4] &&
-	      node->data[5] >= bmin[5] && node->data[5] <= bmax[5]
-	      )
-
-	    pis.Append (node->pi);
-	}
+	if (node->pi)
+	  {
+	    if (
+		node->data[0] >= bmin[0] && node->data[0] <= bmax[0] &&
+		node->data[1] >= bmin[1] && node->data[1] <= bmax[1] &&
+		node->data[2] >= bmin[2] && node->data[2] <= bmax[2] &&
+		node->data[3] >= bmin[3] && node->data[3] <= bmax[3] &&
+		node->data[4] >= bmin[4] && node->data[4] <= bmax[4] &&
+		node->data[5] >= bmin[5] && node->data[5] <= bmax[5]
+		)
+
+	      pis.Append (node->pi);
+	  }
 
       
-      int i1min = (bmin[0] <= node->sep[0]) ? 0 : 1;
-      int i1max = (bmax[0] < node->sep[0]) ? 0 : 1;
-      int i2min = (bmin[1] <= node->sep[1]) ? 0 : 1;
-      int i2max = (bmax[1] < node->sep[1]) ? 0 : 1;
-      int i3min = (bmin[2] <= node->sep[2]) ? 0 : 1;
-      int i3max = (bmax[2] < node->sep[2]) ? 0 : 1;
-
-      int i4min = (bmin[3] <= node->sep[3]) ? 0 : 1;
-      int i4max = (bmax[3] <  node->sep[3]) ? 0 : 1;
-      int i5min = (bmin[4] <= node->sep[4]) ? 0 : 1;
-      int i5max = (bmax[4] <  node->sep[4]) ? 0 : 1;
-      int i6min = (bmin[5] <= node->sep[5]) ? 0 : 1;
-      int i6max = (bmax[5] <  node->sep[5]) ? 0 : 1;
-
-      int i1, i2, i3, i4, i5, i6;
-      for (i1 = i1min; i1 <= i1max; i1++)
-	for (i2 = i2min; i2 <= i2max; i2++)
-	  for (i3 = i3min; i3 <= i3max; i3++)
-	    for (i4 = i4min; i4 <= i4max; i4++)
-	      for (i5 = i5min; i5 <= i5max; i5++)
-		for (i6 = i6min; i6 <= i6max; i6++)
-		  {
-		    i = i1 + 2*i2 + 4*i3 + 8*i4 + 16*i5 +32*i6;
-		    if (node->childs[i])
-		      {
-			stacks++;
-			stack.Elem(stacks) = node->childs[i];
-		      }
-		  }
+	int i1min = (bmin[0] <= node->sep[0]) ? 0 : 1;
+	int i1max = (bmax[0] < node->sep[0]) ? 0 : 1;
+	int i2min = (bmin[1] <= node->sep[1]) ? 0 : 1;
+	int i2max = (bmax[1] < node->sep[1]) ? 0 : 1;
+	int i3min = (bmin[2] <= node->sep[2]) ? 0 : 1;
+	int i3max = (bmax[2] < node->sep[2]) ? 0 : 1;
+
+	int i4min = (bmin[3] <= node->sep[3]) ? 0 : 1;
+	int i4max = (bmax[3] <  node->sep[3]) ? 0 : 1;
+	int i5min = (bmin[4] <= node->sep[4]) ? 0 : 1;
+	int i5max = (bmax[4] <  node->sep[4]) ? 0 : 1;
+	int i6min = (bmin[5] <= node->sep[5]) ? 0 : 1;
+	int i6max = (bmax[5] <  node->sep[5]) ? 0 : 1;
+
+	int i1, i2, i3, i4, i5, i6;
+	for (i1 = i1min; i1 <= i1max; i1++)
+	  for (i2 = i2min; i2 <= i2max; i2++)
+	    for (i3 = i3min; i3 <= i3max; i3++)
+	      for (i4 = i4min; i4 <= i4max; i4++)
+		for (i5 = i5min; i5 <= i5max; i5++)
+		  for (i6 = i6min; i6 <= i6max; i6++)
+		    {
+		      i = i1 + 2*i2 + 4*i3 + 8*i4 + 16*i5 +32*i6;
+		      if (node->childs[i])
+			{
+			  stacks++;
+			  stack.Elem(stacks) = node->childs[i];
+			}
+		    }
       
-      /*
-      if (node->left && bmin[dir] <= node->sep)
-	{
+	/*
+	  if (node->left && bmin[dir] <= node->sep)
+	  {
 	  stacks++;
 	  stack.Elem(stacks) = node->left;
 	  stackdir.Elem(stacks) = ndir;
-	}
-      if (node->right && bmax[dir] >= node->sep)
-	{
+	  }
+	  if (node->right && bmax[dir] >= node->sep)
+	  {
 	  stacks++;
 	  stack.Elem(stacks) = node->right;
 	  stackdir.Elem(stacks) = ndir;
-	}
-      */
-    }
-}
-
-void ADTree6F :: PrintRec (ostream & ost, const ADTreeNode6F * node) const
-{
-  int i;
-  if (node->data)
-    {
-      ost << node->pi << ": ";
-      ost << node->nchilds << " childs, ";
-      for (i = 0; i < 6; i++)
-	ost << node->data[i] << " ";
-      ost << endl;
-    }
-
-  for (i = 0; i < 64; i++)
-    if (node->childs[i])
-      PrintRec (ost, node->childs[i]);
-}
-
+	  }
+	*/
+      }
+  }
 
+  void ADTree6F :: PrintRec (ostream & ost, const ADTreeNode6F * node) const
+  {
+    int i;
+    if (node->data)
+      {
+	ost << node->pi << ": ";
+	ost << node->nchilds << " childs, ";
+	for (i = 0; i < 6; i++)
+	  ost << node->data[i] << " ";
+	ost << endl;
+      }
 
+    for (i = 0; i < 64; i++)
+      if (node->childs[i])
+	PrintRec (ost, node->childs[i]);
+  }
 
-/* ************************************* Point3dTree ********************** */
 
 
+#endif
 
-Point3dTree :: Point3dTree (const Point3d & pmin, const Point3d & pmax)
-{
-  float pmi[3], pma[3];
-  for (int i = 0; i < 3; i++)
-    {
-      pmi[i] = pmin.X(i+1);
-      pma[i] = pmax.X(i+1);
-    }
-  tree = new ADTree3 (pmi, pma);
-}
 
-Point3dTree :: ~Point3dTree ()
-{
-  delete tree;
-}
 
+  /* ************************************* Point3dTree ********************** */
 
 
-void Point3dTree :: Insert (const Point3d & p, int pi)
-{
-  static float pd[3];
-  pd[0] = p.X();
-  pd[1] = p.Y();
-  pd[2] = p.Z();
-  tree->Insert (pd, pi);
-}
 
-void Point3dTree :: GetIntersecting (const Point3d & pmin, const Point3d & pmax, 
-				     ARRAY<int> & pis) const
-{
-  float pmi[3], pma[3];
-  for (int i = 0; i < 3; i++)
-    {
-      pmi[i] = pmin.X(i+1);
-      pma[i] = pmax.X(i+1);
-    }
-  tree->GetIntersecting (pmi, pma, pis);
-}
+  Point3dTree :: Point3dTree (const Point3d & pmin, const Point3d & pmax)
+  {
+    float pmi[3], pma[3];
+    for (int i = 0; i < 3; i++)
+      {
+	pmi[i] = pmin.X(i+1);
+	pma[i] = pmax.X(i+1);
+      }
+    tree = new ADTree3 (pmi, pma);
+  }
+
+  Point3dTree :: ~Point3dTree ()
+  {
+    delete tree;
+  }
+
+
+
+  void Point3dTree :: Insert (const Point3d & p, int pi)
+  {
+    static float pd[3];
+    pd[0] = p.X();
+    pd[1] = p.Y();
+    pd[2] = p.Z();
+    tree->Insert (pd, pi);
+  }
+
+  void Point3dTree :: GetIntersecting (const Point3d & pmin, const Point3d & pmax, 
+				       ARRAY<int> & pis) const
+  {
+    float pmi[3], pma[3];
+    for (int i = 0; i < 3; i++)
+      {
+	pmi[i] = pmin.X(i+1);
+	pma[i] = pmax.X(i+1);
+      }
+    tree->GetIntersecting (pmi, pma, pis);
+  }
 
 
 
@@ -2240,53 +2193,53 @@ void Point3dTree :: GetIntersecting (const Point3d & pmin, const Point3d & pmax,
 
 
 
-Box3dTree :: Box3dTree (const Point3d & apmin, const Point3d & apmax)
-{
-  boxpmin = apmin;
-  boxpmax = apmax;
-  float tpmin[6], tpmax[6];
-  for (int i = 0; i < 3; i++)
-    {
-      tpmin[i] = tpmin[i+3] = boxpmin.X(i+1);
-      tpmax[i] = tpmax[i+3] = boxpmax.X(i+1);
-    }
-  tree = new ADTree6 (tpmin, tpmax);
-}
+  Box3dTree :: Box3dTree (const Point3d & apmin, const Point3d & apmax)
+  {
+    boxpmin = apmin;
+    boxpmax = apmax;
+    float tpmin[6], tpmax[6];
+    for (int i = 0; i < 3; i++)
+      {
+	tpmin[i] = tpmin[i+3] = boxpmin.X(i+1);
+	tpmax[i] = tpmax[i+3] = boxpmax.X(i+1);
+      }
+    tree = new ADTree6 (tpmin, tpmax);
+  }
 
-Box3dTree :: ~Box3dTree ()
-{
-  delete tree;
-}
+  Box3dTree :: ~Box3dTree ()
+  {
+    delete tree;
+  }
 
-void Box3dTree :: Insert (const Point3d & bmin, const Point3d & bmax, int pi)
-{
-  static float tp[6];
+  void Box3dTree :: Insert (const Point3d & bmin, const Point3d & bmax, int pi)
+  {
+    static float tp[6];
 
-  for (int i = 0; i < 3; i++)
-    {
-      tp[i] = bmin.X(i+1);
-      tp[i+3] = bmax.X(i+1);
-    }
+    for (int i = 0; i < 3; i++)
+      {
+	tp[i] = bmin.X(i+1);
+	tp[i+3] = bmax.X(i+1);
+      }
 
-  tree->Insert (tp, pi);
-}
+    tree->Insert (tp, pi);
+  }
 
-void Box3dTree ::GetIntersecting (const Point3d & pmin, const Point3d & pmax, 
-				  ARRAY<int> & pis) const
-{
-  float tpmin[6];
-  float tpmax[6];
+  void Box3dTree ::GetIntersecting (const Point3d & pmin, const Point3d & pmax, 
+				    ARRAY<int> & pis) const
+  {
+    float tpmin[6];
+    float tpmax[6];
 
-  for (int i = 0; i < 3; i++)
-    {
-      tpmin[i] = boxpmin.X(i+1);
-      tpmax[i] = pmax.X(i+1);
+    for (int i = 0; i < 3; i++)
+      {
+	tpmin[i] = boxpmin.X(i+1);
+	tpmax[i] = pmax.X(i+1);
       
-      tpmin[i+3] = pmin.X(i+1);
-      tpmax[i+3] = boxpmax.X(i+1);
-    }
+	tpmin[i+3] = pmin.X(i+1);
+	tpmax[i+3] = boxpmax.X(i+1);
+      }
 
-  tree->GetIntersecting (tpmin, tpmax, pis);
-}
+    tree->GetIntersecting (tpmin, tpmax, pis);
+  }
 
 }
diff --git a/Netgen/libsrc/gprim/adtree.hpp b/Netgen/libsrc/gprim/adtree.hpp
index d28f873086..64f3509e1a 100644
--- a/Netgen/libsrc/gprim/adtree.hpp
+++ b/Netgen/libsrc/gprim/adtree.hpp
@@ -124,7 +124,7 @@ public:
 };
 
 
-
+/*
 
 // divide each direction
 #define ADTN_DIV 10
@@ -320,7 +320,7 @@ public:
 
 
 
-
+*/
 
 
 
@@ -380,7 +380,7 @@ public:
 
 
 
-
+/*
 
 class ADTreeNode6F
 {
@@ -436,7 +436,7 @@ public:
 
 
 
-
+*/
 
 
 
diff --git a/Netgen/libsrc/gprim/geom3d.cpp b/Netgen/libsrc/gprim/geom3d.cpp
index b243e62c3a..eaee776778 100644
--- a/Netgen/libsrc/gprim/geom3d.cpp
+++ b/Netgen/libsrc/gprim/geom3d.cpp
@@ -1,3 +1,4 @@
+#include <algorithm>
 #include <mystdlib.h>
 
 #include <myadt.hpp>
@@ -175,6 +176,15 @@ Box3d :: Box3d ( const Box3d & b2 )
     }
 }
 
+Box3d :: Box3d ( const Box<3> & b2 )
+{
+  for (int i = 0; i < 3; i++)
+    {
+      minx[i] = b2.PMin()(i);
+      maxx[i] = b2.PMax()(i);
+    }
+}
+
 
 /*
 int Box3d :: Intersect (const Box3d & box2) const
diff --git a/Netgen/libsrc/gprim/geom3d.hpp b/Netgen/libsrc/gprim/geom3d.hpp
index fd60e6b26f..56bff4ad6b 100644
--- a/Netgen/libsrc/gprim/geom3d.hpp
+++ b/Netgen/libsrc/gprim/geom3d.hpp
@@ -417,11 +417,13 @@ inline Vec3d & Vec3d :: operator/= (double s)
       x[1] /= s;
       x[2] /= s;
     }
+#ifdef DEBUG
   else
     {
       cerr << "Vec div by 0, v = " << (*this) << endl;
       //      MyError ("Vec3d::operator /=: Divisioin by zero");
     }
+#endif
   return *this;
 }
 
@@ -571,6 +573,8 @@ public:
   ///
   Box3d (const Point3d& p1, const Point3d& p2);
   ///
+  Box3d (const Box<3> & b2);
+  ///
   double MinX () const { return minx[0]; }
   ///
   double MaxX () const { return maxx[0]; }
diff --git a/Netgen/libsrc/gprim/geomfuncs.hpp b/Netgen/libsrc/gprim/geomfuncs.hpp
index b013261df8..b9228c8583 100644
--- a/Netgen/libsrc/gprim/geomfuncs.hpp
+++ b/Netgen/libsrc/gprim/geomfuncs.hpp
@@ -115,8 +115,15 @@ inline void CalcInverse (const Mat<2,3> & m, Mat<3,2> & inv)
   inv = Trans (m) * ainv;
 }
 
-// void CalcInverse (const Mat<3,2> & m, Mat<2,3> & inv);
+void CalcInverse (const Mat<3,2> & m, Mat<2,3> & inv);
 
+inline void CalcInverse (const Mat<3,2> & m, Mat<2,3> & inv)
+{
+  Mat<2,2> a = Trans (m) * m;
+  Mat<2,2> ainv;
+  CalcInverse (a, ainv);
+  inv = ainv * Trans (m);
+}
 
 
 double Det (const Mat<2,2> & m);
diff --git a/Netgen/libsrc/gprim/geomobjects.hpp b/Netgen/libsrc/gprim/geomobjects.hpp
index 9f6a8f025b..2db82cf9e9 100644
--- a/Netgen/libsrc/gprim/geomobjects.hpp
+++ b/Netgen/libsrc/gprim/geomobjects.hpp
@@ -74,6 +74,10 @@ public:
   explicit Vec (const Point<D> & p)
   { for (int i = 0; i < D; i++) x[i] = p(i); }
 
+  Vec (const Vec<D> & p1, const Vec<D> & p2)
+  { for(int i=0; i<D; i++) x[i] = p2(i)-p1(1); }
+  
+
 
   Vec & operator= (const Vec<D> & p2)
   {
@@ -150,6 +154,8 @@ public:
 
   double & operator() (int i, int j) { return x[i*W+j]; }
   const double & operator() (int i, int j) const { return x[i*W+j]; }
+  double & operator() (int i) { return x[i]; }
+  const double & operator() (int i) const { return x[i]; }
 
   Vec<H> Col (int i) const
   {
@@ -273,6 +279,13 @@ protected:
 public:
   ///
   BoxSphere () { };
+ ///
+  BoxSphere (const Box<D> & box) 
+  : Box<D> (box) 
+  { 
+    CalcDiamCenter();
+  };
+
   ///
   BoxSphere ( Point<D> pmin, Point<D> pmax )
     : Box<D> (pmin, pmax)
diff --git a/Netgen/libsrc/gprim/geomops.hpp b/Netgen/libsrc/gprim/geomops.hpp
index 0b783c076b..755f35a878 100644
--- a/Netgen/libsrc/gprim/geomops.hpp
+++ b/Netgen/libsrc/gprim/geomops.hpp
@@ -287,6 +287,23 @@ inline Mat<3,2> operator* (const Mat<3,2> & a, const Mat<2,2> & b)
   return m;
 }
 
+
+
+inline Mat<2,3> operator* (const Mat<2,2> & a, const Mat<2,3> & b)
+{
+  Mat<2,3> m;
+  for (int i = 0; i < 2; i++)
+    for (int j = 0; j < 3; j++)
+      {
+	double sum = 0;
+	for (int k = 0; k < 2; k++)
+	  sum += a(i,k) * b(k, j);
+	m(i,j) = sum; 
+      }
+  return m;
+}
+
+
 inline Mat<3,3> operator* (const Mat<3,3> & a, const Mat<3,3> & b)
 {
   Mat<3,3> m;
diff --git a/Netgen/libsrc/gprim/transform3d.cpp b/Netgen/libsrc/gprim/transform3d.cpp
index 40ef4e3b2a..ea62fffe5a 100644
--- a/Netgen/libsrc/gprim/transform3d.cpp
+++ b/Netgen/libsrc/gprim/transform3d.cpp
@@ -54,6 +54,7 @@ Transformation3d (const Point3d & c, double alpha,
   ht.Combine (ht2, r1);
   Combine (ht, tcinv);
 
+ cout << "Rotation - Transformation:" << (*this) << endl;
   //  (*testout) << "Rotation - Transformation:" << (*this) << endl;
 }
 
diff --git a/Netgen/libsrc/gprim/transform3d.hpp b/Netgen/libsrc/gprim/transform3d.hpp
index 9eccc5a7f9..7472257297 100644
--- a/Netgen/libsrc/gprim/transform3d.hpp
+++ b/Netgen/libsrc/gprim/transform3d.hpp
@@ -93,7 +93,27 @@ public:
   // rotation with ...
   Transformation (const Point<D> & c, double alpha, double beta, double gamma)
   {
-    ;
+    // total = T_c x Rot_0 x T_c^{-1}
+    // Use Euler angles, see many books from tech mech, e.g. 
+    // Shabana "multibody systems"
+    
+    Vec<D> vc(c);
+    Transformation<D> tc(vc);
+    Transformation<D> tcinv(-vc);
+    // tc.CalcInverse (tcinv);
+    
+    Transformation<D> r1, r2, r3, ht, ht2;
+    r1.SetAxisRotation (3, alpha);
+    r2.SetAxisRotation (1, beta);
+    r3.SetAxisRotation (3, gamma);
+    
+    ht.Combine (tc, r3);
+    ht2.Combine (ht, r2);
+    ht.Combine (ht2, r1);
+    Combine (ht, tcinv);
+    
+    // cout << "Rotation - Transformation:" << (*this) << endl;
+    //  (*testout) << "Rotation - Transformation:" << (*this) << endl;
   }
 
   /// 
@@ -106,8 +126,31 @@ public:
     m = ta.m * tb.m;
   }
 
+
+
   /// dir = 1..3 (== x..z)
-  void SetAxisRotation (int dir, double alpha);
+  void SetAxisRotation (int dir, double alpha)
+  {
+    double co = cos(alpha);
+    double si = sin(alpha);
+    dir--;
+    int pos1 = (dir+1) % 3;
+    int pos2 = (dir+2) % 3;
+    
+    int i, j;
+    for (i = 0; i <= 2; i++)
+    {
+      v(i) = 0;
+      for (j = 0; j <= 2; j++)
+	m(i,j) = 0;
+    }
+    
+    m(dir,dir) = 1;
+    m(pos1, pos1) = co;
+    m(pos2, pos2) = co;
+    m(pos1, pos2) = si;
+    m(pos2, pos1) = -si;
+  }
 
   ///
   void Transform (const Point<D> & from, Point<D> & to) const
diff --git a/Netgen/libsrc/include/FlexLexer.h b/Netgen/libsrc/include/FlexLexer.h
index abc6b53625..4b9b53cd13 100644
--- a/Netgen/libsrc/include/FlexLexer.h
+++ b/Netgen/libsrc/include/FlexLexer.h
@@ -1,4 +1,4 @@
-// $Header: /cvsroot/gmsh/Netgen/libsrc/include/FlexLexer.h,v 1.1 2004-06-26 18:03:24 geuzaine Exp $
+// $Header: /cvsroot/gmsh/Netgen/libsrc/include/FlexLexer.h,v 1.2 2004-12-08 20:01:21 geuzaine Exp $
 
 // FlexLexer.h -- define interfaces for lexical analyzer classes generated
 //		  by flex
diff --git a/Netgen/libsrc/include/mystdlib.h b/Netgen/libsrc/include/mystdlib.h
index 8c5544407c..2249aea3a2 100644
--- a/Netgen/libsrc/include/mystdlib.h
+++ b/Netgen/libsrc/include/mystdlib.h
@@ -31,6 +31,7 @@
 
 #include <new>
 #include <string>
+#include <typeinfo>
 
 #ifndef M_PI
 #define M_PI 3.14159265358979323846
diff --git a/Netgen/libsrc/interface/Makefile b/Netgen/libsrc/interface/Makefile
index 98d943a028..c47dbe2c5a 100644
--- a/Netgen/libsrc/interface/Makefile
+++ b/Netgen/libsrc/interface/Makefile
@@ -1,4 +1,4 @@
-src = nginterface.cpp writeuser.cpp writediffpack.cpp writeabaqus.cpp writefluent.cpp writepermas.cpp writetochnog.cpp writetecplot.cpp wuchemnitz.cpp writetochnog.cpp writefeap.cpp readuser.cpp importsolution.cpp
+src = nginterface.cpp writeuser.cpp writediffpack.cpp writeabaqus.cpp writefluent.cpp writepermas.cpp writetochnog.cpp writetecplot.cpp wuchemnitz.cpp writetochnog.cpp writefeap.cpp writeelmer.cpp  writegmsh.cpp readuser.cpp importsolution.cpp 
 #
 lib = nginterface
 libpath = libsrc/interface
diff --git a/Netgen/libsrc/interface/nginterface.cpp b/Netgen/libsrc/interface/nginterface.cpp
index 5510f28a27..34182721ea 100644
--- a/Netgen/libsrc/interface/nginterface.cpp
+++ b/Netgen/libsrc/interface/nginterface.cpp
@@ -6,12 +6,18 @@
 #include <geometry2d.hpp>
 #include <stlgeom.hpp>
 
-#include <visual.hpp>
+#ifdef OCCGEOMETRY
+#include <occgeom.hpp>
+#endif
+
+
+//#include <visual.hpp>
 
 #include "nginterface.h"
 // #include <FlexLexer.h>
 
 
+// #include <mystdlib.h>
 
 
 namespace netgen
@@ -20,12 +26,16 @@ namespace netgen
   extern VisualSceneMesh vsmesh;
   extern Tcl_Interp * tcl_interp;
 
-  extern SplineGeometry2d * geometry2d;
-  extern CSGeometry * parsegeom;
-  extern CSGeometry * geometry;
+  extern AutoPtr<SplineGeometry2d> geometry2d;
+  extern AutoPtr<CSGeometry> geometry;
   extern STLGeometry * stlgeometry;
+#ifdef OCCGEOMETRY
+  extern OCCGeometry * occgeometry;
+#endif
 
+#ifdef OPENGL
   extern VisualSceneSolution vssolution;
+#endif
   extern CSGeometry * ParseCSG (istream & istr);
 }
 
@@ -47,50 +57,81 @@ void Ng_LoadGeometry (char * filename)
 {
   ifstream infile (filename);
 
-  if (geometry) 
-    delete geometry;
+  geometry.Reset ();
+  geometry2d.Reset ();
 
-  if (strcmp (&filename[strlen(filename)-3], "geo") == 0)
-    {
-      /*
-      geometry = new CSGeometry(filename);
-      
-      parsegeom = geometry;
-      
-      lexer = new yyFlexLexer (&infile);
-      
-      extern int yyparse ();
-      yyparse ();
-      delete lexer; 
-      */
+#ifdef OCCGEOMETRY
+  delete occgeometry;
+  occgeometry = 0;
+#endif
 
-      geometry = netgen::ParseCSG (infile);
+  if ((strcmp (&filename[strlen(filename)-3], "geo") == 0) ||
+      (strcmp (&filename[strlen(filename)-3], "GEO") == 0) ||
+      (strcmp (&filename[strlen(filename)-3], "Geo") == 0))
+    {
+      geometry.Reset (netgen::ParseCSG (infile));
 
       if (!geometry)
-	throw NgException ("input file not found");
+	{
+	  geometry.Reset (new CSGeometry ());
+	  throw NgException ("input file not found");
+	}
 
       geometry -> FindIdenticSurfaces(1e-6);
+
+      double detail = atof (Tcl_GetVar (tcl_interp, "geooptions.detail", 0));
+      double facets = atof (Tcl_GetVar (tcl_interp, "geooptions.facets", 0));
       Box<3> box (geometry->BoundingBox());
+      
+      if (atoi (Tcl_GetVar (tcl_interp, "geooptions.drawcsg", 0)))
+	geometry->CalcTriangleApproximation(box, detail, facets);
 
-      geometry->CalcTriangleApproximation (box, 0.01, 10);
+      //      geometry->CalcTriangleApproximation (box, 0.01, 10);
     }
-  else
-    {
-      geometry = new CSGeometry("");
 
-      if (strcmp (&filename[strlen(filename)-4], "in2d") == 0)
-	{
+  else if (strcmp (&filename[strlen(filename)-4], "in2d") == 0)
+    {
+      geometry2d.Reset (new SplineGeometry2d());
+      geometry2d -> Load (filename);
+    }
 
-	  if (geometry2d)
-	    delete geometry2d;
-	  geometry2d = new SplineGeometry2d();
-	  geometry2d -> Load (filename);
-	}
+  else if ((strcmp (&filename[strlen(filename)-3], "stl") == 0) ||
+	   (strcmp (&filename[strlen(filename)-3], "STL") == 0) ||
+	   (strcmp (&filename[strlen(filename)-3], "Stl") == 0))
+    {
+      ifstream infile(filename);
+      stlgeometry = STLGeometry :: Load (infile);
+      stlgeometry->edgesfound = 0;
+      Mesh meshdummy;
+      stlgeometry->Clear();
+      stlgeometry->BuildEdges();
+      stlgeometry->MakeAtlas(meshdummy);
+      stlgeometry->CalcFaceNums();
+      stlgeometry->AddFaceEdges();
+      stlgeometry->LinkEdges();
+    }
 
-      else
-	{
-	  cerr << "Unknown geometry extension!!" << endl;
-	}
+#ifdef OCCGEOMETRY
+  else if ((strcmp (&filename[strlen(filename)-4], "iges") == 0) ||
+	   (strcmp (&filename[strlen(filename)-3], "igs") == 0) ||
+	   (strcmp (&filename[strlen(filename)-3], "IGS") == 0) ||
+	   (strcmp (&filename[strlen(filename)-4], "IGES") == 0))
+    {
+      PrintMessage (1, "Load IGES geometry file ", filename);
+      occgeometry = LoadOCC_IGES (filename);
+    }
+  else if ((strcmp (&filename[strlen(filename)-4], "step") == 0) ||
+	   (strcmp (&filename[strlen(filename)-3], "stp") == 0) ||
+	   (strcmp (&filename[strlen(filename)-3], "STP") == 0) ||
+	   (strcmp (&filename[strlen(filename)-4], "STEP") == 0))
+    {
+      PrintMessage (1, "Load STEP geometry file ", filename);
+      occgeometry = LoadOCC_STEP (filename);
+    }
+#endif
+  else
+    {
+      cerr << "Unknown geometry extension!!" << endl;
     }
 }                          
 
@@ -293,7 +334,7 @@ NG_ELEMENT_TYPE Ng_GetSurfaceElement (int ei, int * epi, int * np)
     {
       const Segment & seg = mesh->LineSegment (ei);
 
-      if (!seg.pmid)
+      if (seg.pmid < 0)
 	{
 	  epi[0] = seg.p1;
 	  epi[1] = seg.p2;
@@ -329,6 +370,8 @@ void Ng_GetNormalVector (int sei, int locpi, double * nv)
   nv[0] = 0; 
   nv[1] = 0;
   nv[2] = 1;
+
+  (*testout) << "Ng_GetNormalVector (sei = " << sei << ", locpi = " << locpi << ")" << endl;
   
   if (mesh->GetDimension() == 3)
     {
@@ -337,9 +380,24 @@ void Ng_GetNormalVector (int sei, int locpi, double * nv)
       p = mesh->Point (mesh->SurfaceElement(sei).PNum(locpi));
 
       int surfi = mesh->GetFaceDescriptor(mesh->SurfaceElement(sei).GetIndex()).SurfNr();
+
+      (*testout) << "surfi = " << surfi << endl;
+#ifdef OCCGEOMETRY
+      if (occgeometry)
+	{
+	  PointGeomInfo gi = mesh->SurfaceElement(sei).GeomInfoPi(locpi);
+	  occgeometry->GetSurface (surfi).GetNormalVector(p, gi, n);
+	  nv[0] = n(0);
+	  nv[1] = n(1);
+	  nv[2] = n(2);
+	}
+      else
+#endif
       if (geometry)
 	{
-	  geometry->GetSurface (surfi) -> GetNormalVector(p, n);
+	  (*testout) << "geometry defined" << endl;
+	  n = geometry->GetSurface (surfi) -> GetNormalVector(p);
+	  (*testout) << "aus is" << endl;
 	  nv[0] = n(0);
 	  nv[1] = n(1);
 	  nv[2] = n(2);
@@ -348,13 +406,13 @@ void Ng_GetNormalVector (int sei, int locpi, double * nv)
 }
 
 
-int Ng_FindElementOfPoint (double * p, double * lami)
+int Ng_FindElementOfPoint (double * p, double * lami, int build_searchtree, int index)
 {
   if (mesh->GetDimension() == 3)
     {
       Point3d p3d(p[0], p[1], p[2]);
       int ind = 
-	mesh->GetElementOfPoint(p3d, lami);
+	mesh->GetElementOfPoint(p3d, lami, build_searchtree != 0, index);
       return ind;
     }
   else
@@ -362,7 +420,7 @@ int Ng_FindElementOfPoint (double * p, double * lami)
       double lam3[3];
       Point3d p2d(p[0], p[1], 0);
       int ind = 
-	mesh->GetElementOfPoint(p2d, lam3);
+	mesh->GetElementOfPoint(p2d, lam3, build_searchtree != 0, index);
       lami[0] = lam3[0];
       lami[1] = lam3[1];
       return ind;
@@ -382,7 +440,6 @@ void Ng_GetElementTransformation (int ei, const double * xi,
 
       mesh->GetCurvedElements().CalcSurfaceTransformation (xl, ei-1, xg, dx);
 
-      // still 1-based arrays
       if (x)
 	{
 	  for (int i = 0; i < 2; i++)
@@ -471,6 +528,12 @@ void Ng_GetSurfaceElementTransformation (int sei, const double * xi,
 
 
 
+void Ng_GetSurfaceElementNeighbouringDomains(const int selnr, int & in, int & out)
+{
+  in = mesh->GetFaceDescriptor((*mesh)[static_cast<SurfaceElementIndex>(selnr)].GetIndex()).DomainIn();
+  out = mesh->GetFaceDescriptor((*mesh)[static_cast<SurfaceElementIndex>(selnr)].GetIndex()).DomainOut();
+}
+
 
 void Ng_SetRefinementFlag (int ei, int flag)
 {
@@ -491,33 +554,34 @@ void Ng_Refine (NG_REFINEMENT_TYPE reftype)
 {
   BisectionOptions biopt;
   biopt.usemarkedelements = 1;
+  biopt.refine_p = 0;
   biopt.refine_hp = 0;
-  if (reftype == NG_REFINE_P || reftype == NG_REFINE_HP)
+  if (reftype == NG_REFINE_P)
+    biopt.refine_p = 1;
+  if (reftype == NG_REFINE_HP)
     biopt.refine_hp = 1;
+  Refinement * ref;
 
-  if (geometry && mesh->GetDimension() == 3)
-    {
-      RefinementSurfaces ref (*geometry);
-      ref.Bisect (*mesh, biopt);
-    }
+  if (geometry2d)
+    ref = new Refinement2d(*geometry2d);
   else if (stlgeometry)
-    {
-      RefinementSTLGeometry ref (*stlgeometry);
-      ref.Bisect (*mesh, biopt);
-    }
-  else if (geometry2d)
-    {
-      Refinement2d ref (*geometry2d);
-      ref.Bisect (*mesh, biopt);
-    }
+    ref = new RefinementSTLGeometry(*stlgeometry);
+#ifdef OCCGEOMETRY
+  else if (occgeometry)
+    ref = new OCCRefinementSurfaces (*occgeometry);
+#endif
+  else if (geometry && mesh->GetDimension() == 3)
+    ref = new RefinementSurfaces(*geometry);
   else
     {
-      cout << "No geometry available" << endl;
-      Refinement ref;
-      ref.Bisect (*mesh, biopt);
+      ref = new Refinement();
     }
 
+  ref -> Bisect (*mesh, biopt);
+
   mesh -> UpdateTopology();
+  // mesh -> GetCurvedElements().BuildCurvedElements (ref, mparam.elementorder);
+  delete ref;
 }
 
 void Ng_SecondOrder ()
@@ -552,7 +616,17 @@ void Ng_SecondOrder ()
 
 void Ng_HPRefinement (int levels)
 {
-  HPRefinement (*mesh, levels);
+  Refinement * ref;
+
+  if (stlgeometry)
+    ref = new RefinementSTLGeometry (*stlgeometry);
+  else if (geometry2d)
+    ref = new Refinement2d (*geometry2d);
+  else
+    ref = new RefinementSurfaces (*geometry);
+
+
+  HPRefinement (*mesh, ref, levels);
 }
 
 
@@ -562,6 +636,10 @@ void Ng_HighOrder (int order)
 
   if (stlgeometry)
     ref = new RefinementSTLGeometry (*stlgeometry);
+#ifdef OCCGEOMETRY
+  else if (occgeometry)
+    ref = new OCCRefinementSurfaces (*occgeometry);
+#endif
   else if (geometry2d)
     ref = new Refinement2d (*geometry2d);
   else
@@ -1019,6 +1097,26 @@ void Ng_GetEdge_Vertices (int ednr, int * vert)
 }
 
 
+int Ng_GetNVertexElements (int vnr)
+{
+  if (mesh->GetDimension() == 3)
+    return mesh->GetTopology().GetVertexElements(vnr).Size();
+  else
+    return mesh->GetTopology().GetVertexSurfaceElements(vnr).Size();
+}
+
+void Ng_GetVertexElements (int vnr, int * els)
+{
+  FlatArray<int> ia(0,0);
+  if (mesh->GetDimension() == 3)
+    ia = mesh->GetTopology().GetVertexElements(vnr);
+  else
+    ia = mesh->GetTopology().GetVertexSurfaceElements(vnr);
+  for (int i = 0; i < ia.Size(); i++)
+    els[i] = ia[i];
+}
+
+
 int Ng_GetElementOrder (int enr)
 {
   if (mesh->GetDimension() == 3)
@@ -1122,6 +1220,7 @@ void Ng_InitSolutionData (Ng_SolutionData * soldata)
 
 void Ng_SetSolutionData (Ng_SolutionData * soldata)
 {
+#ifdef OPENGL
   //   vssolution.ClearSolutionData ();
   VisualSceneSolution::SolData * vss = new VisualSceneSolution::SolData;
 
@@ -1140,24 +1239,35 @@ void Ng_SetSolutionData (Ng_SolutionData * soldata)
   vss->soltype = VisualSceneSolution::SolType (soldata->soltype);
   vss->solclass = soldata->solclass;
   vssolution.AddSolutionData (vss);
+#endif
 }
 
+void Ng_ClearSolutionData ()
+{
+  vssolution.ClearSolutionData();
+}
+
+
 
 void Ng_Redraw ()
 {
+#ifdef OPENGL
   vssolution.UpdateSolutionTimeStamp();
   Render();
+#endif
 }
 
 
 void Ng_SetVisualizationParameter (const char * name, const char * value)
 {
+#ifdef OPENGL
   char buf[100];
   sprintf (buf, "visoptions.%s", name);
   cout << "name = " << name << ", value = " << value << endl;
   cout << "set tcl-variable " << buf << " to " << value << endl;
   Tcl_SetVar (tcl_interp, buf, const_cast<char*> (value), 0);
   Tcl_Eval (tcl_interp, "Ng_Vis_Set parameters;");
+#endif
 }
 
 
@@ -1198,7 +1308,7 @@ void PlayAnimFile(const char* name, int speed, int maxcnt)
   for (i = 1; i <= ne; i++)
     {
       int j;
-      Element2d tri(3);
+      Element2d tri(TRIG);
       tri.SetIndex(1); //faceind
       
       for (j = 1; j <= 3; j++)
@@ -1240,62 +1350,127 @@ void Ng_GetPeriodicVertices (int * pairs)
       pairs[2*i+1] = apairs[i].I2();
     }
       
-	/*<<<<<< nginterface.cpp
-      infile >> np;
-      for (i = 1; i <= np; i++)
+}
+
+
+
+int Ng_GetNPeriodicEdges ()
+{
+  ARRAY<INDEX,PointIndex::BASE> map;
+  const MeshTopology & top = mesh->GetTopology();
+  int nse = mesh->GetNSeg();
+
+  int cnt = 0;
+  //  for (int id = 1; id <= mesh->GetIdentifications().GetMaxNr(); id++)
+    {
+      mesh->GetIdentifications().GetMap(0, map);
+      //(*testout) << "ident-map " << id << ":" << endl << map << endl;
+
+      for (SegmentIndex si = 0; si < nse; si++)
 	{
-	  Point3d p;
-	  infile >> p.X() >> p.Y() >> p.Z();
-	  if (firsttime)
-	    mesh->AddPoint (p);
-	  else
-	    mesh->Point(i)=p;
+	  PointIndex other1 = map[(*mesh)[si].p1];
+	  PointIndex other2 = map[(*mesh)[si].p2];
+	  //  (*testout) << "seg = " << (*mesh)[si] << "; other = " 
+	  //     << other1 << "-" << other2 << endl;
+	  if (other1 && other2 && mesh->IsSegment (other1, other2))
+	    {
+	      cnt++;
+	    }
 	}
+    }
+  return cnt;
+}
 
-      //firsttime = 0;
-      Ng_Redraw();
-      //}
+void Ng_GetPeriodicEdges (int * pairs)
+{
+  ARRAY<INDEX,PointIndex::BASE> map;
+  const MeshTopology & top = mesh->GetTopology();
+  int nse = mesh->GetNSeg();
+
+  int cnt = 0;
+  //  for (int id = 1; id <= mesh->GetIdentifications().GetMaxNr(); id++)
+    {
+      mesh->GetIdentifications().GetMap(0, map);
+      
+      //(*testout) << "map = " << map << endl;
+
+      for (SegmentIndex si = 0; si < nse; si++)
+	{
+	  PointIndex other1 = map[(*mesh)[si].p1];
+	  PointIndex other2 = map[(*mesh)[si].p2];
+	  if (other1 && other2 && mesh->IsSegment (other1, other2))
+	    {
+	      SegmentIndex otherseg = mesh->SegmentNr (other1, other2);
+	      pairs[cnt++] = top.GetSegmentEdge (si+1);
+	      pairs[cnt++] = top.GetSegmentEdge (otherseg+1);
+	    }
+	}
+    }
 }
 
-		
-int Ng_GetNPeriodicVertices ()
+
+
+void Ng_PushStatus (const char * str)
 {
-  ARRAY<INDEX_2> apairs;
-  mesh->GetIdentifications().GetPairs (0, apairs);
-  return apairs.Size();
+  PushStatus (MyStr (str));
 }
 
+void Ng_PopStatus ()
+{
+  PopStatus ();
+}
 
-// pairs should be an integer array of 2*npairs
-void Ng_GetPeriodicVertices (int * pairs)
+void Ng_SetThreadPercentage (double percent)
 {
-  ARRAY<INDEX_2> apairs;
-  mesh->GetIdentifications().GetPairs (0, apairs);
-  for (int i = 0; i < apairs.Size(); i++)
-    {
-      pairs[2*i] = apairs[i].I1();
-      pairs[2*i+1] = apairs[i].I2();
-    }
-=======
-      if (firsttime)
-	mesh->AddSurfaceElement (tri);
-    }
+  SetThreadPercent (percent);
+}
+
+
+///// Added by Roman Stainko ....
+int Ng_GetVertex_Elements( int vnr, int* elems )
+{
+  const MeshTopology& topology = mesh->GetTopology();
+  ArrayMem<int,4> indexArray;
+  topology.GetVertexElements( vnr, indexArray );
   
-  infile >> np;
-  for (i = 1; i <= np; i++)
-    {
-      Point3d p;
-      infile >> p.X() >> p.Y() >> p.Z();
-      if (firsttime)
-	mesh->AddPoint (p);
-      else
-	mesh->Point(i)=p;
-    }
+  for( int i=0; i<indexArray.Size(); i++ )
+    elems[i] = indexArray[i];
   
-  //firsttime = 0;
-  Ng_Redraw();
->>>>>>> 1.9 
-*/
+  return indexArray.Size();
+}
+
+///// Added by Roman Stainko ....
+int Ng_GetVertex_SurfaceElements( int vnr, int* elems )
+{
+  const MeshTopology& topology = mesh->GetTopology();
+  ArrayMem<int,4> indexArray;
+  topology.GetVertexSurfaceElements( vnr, indexArray );
+  
+  for( int i=0; i<indexArray.Size(); i++ )
+    elems[i] = indexArray[i];
+  
+  return indexArray.Size();
 }
 
+///// Added by Roman Stainko ....
+int Ng_GetVertex_NElements( int vnr )
+{
+  const MeshTopology& topology = mesh->GetTopology();
+  ArrayMem<int,4> indexArray;
+  topology.GetVertexElements( vnr, indexArray );
+  
+  return indexArray.Size();
+}
+
+///// Added by Roman Stainko ....
+int Ng_GetVertex_NSurfaceElements( int vnr )
+{
+  const MeshTopology& topology = mesh->GetTopology();
+  ArrayMem<int,4> indexArray;
+  topology.GetVertexSurfaceElements( vnr, indexArray );
+
+  return indexArray.Size();
+}
+
+
 
diff --git a/Netgen/libsrc/interface/nginterface.h b/Netgen/libsrc/interface/nginterface.h
index e7949900a0..dcb8ddd332 100644
--- a/Netgen/libsrc/interface/nginterface.h
+++ b/Netgen/libsrc/interface/nginterface.h
@@ -19,7 +19,7 @@
 #define NG_ELEMENT_MAXPOINTS 12
 
 // max number of nodes per surface element
-#define NG_SURFACE_ELEMENT_MAXPOINTS 6
+#define NG_SURFACE_ELEMENT_MAXPOINTS 8
 
 
 
@@ -89,7 +89,8 @@ extern "C" {
   
 
   // Find element of point, returns local coordinates
-  int Ng_FindElementOfPoint (double * p, double * lami);
+  int Ng_FindElementOfPoint (double * p, double * lami,
+			     int build_searchtrees = 0, int index = -1);
   
 
   /// Curved Elemens:
@@ -140,10 +141,15 @@ extern "C" {
   int Ng_GetSurfaceElement_Edges (int selnr, int * edges, int * orient = 0);
   int Ng_GetSurfaceElement_Face (int selnr, int * orient = 0);
 
+  void Ng_GetSurfaceElementNeighbouringDomains(const int selnr, int & in, int & out);
+       
   int Ng_GetFace_Vertices (int fnr, int * vert);
   void Ng_GetEdge_Vertices (int ednr, int * vert);
   int Ng_GetFace_Edges (int fnr, int * edge);
 
+  int Ng_GetNVertexElements (int vnr);
+  void Ng_GetVertexElements (int vnr, int * els);
+
   int Ng_GetElementOrder (int enr);
 
   // Multilevel functions:
@@ -202,6 +208,8 @@ namespace netgen {
   void Ng_InitSolutionData (Ng_SolutionData * soldata);
   // set solution data
   void Ng_SetSolutionData (Ng_SolutionData * soldata);
+  /// delete gridfunctions
+  void Ng_ClearSolutionData();
   // redraw 
   void Ng_Redraw();
   //
@@ -212,9 +220,24 @@ namespace netgen {
   // number of periodic vertices  
   int Ng_GetNPeriodicVertices ();
   // pairs should be an integer array of 2*npairs
-  void Ng_GetPeriodicVertices (int * pairs);
+  void Ng_GetPeriodicVertices (int * pairs); 
+
+  // number of periodic edges  
+  int Ng_GetNPeriodicEdges ();
+  // pairs should be an integer array of 2*npairs
+  void Ng_GetPeriodicEdges (int * pairs); 
 
 
+  void Ng_PushStatus (const char * str);
+  void Ng_PopStatus ();
+  void Ng_SetThreadPercentage (double percent);
+  
+  //// added by Roman Stainko ....
+  int Ng_GetVertex_Elements( int vnr, int* elems);
+  int Ng_GetVertex_SurfaceElements( int vnr, int* elems );
+  int Ng_GetVertex_NElements( int vnr );
+  int Ng_GetVertex_NSurfaceElements( int vnr );
+
 #ifdef __cplusplus
 }
 #endif
diff --git a/Netgen/libsrc/interface/nglib.cpp b/Netgen/libsrc/interface/nglib.cpp
index d309cd7189..d2e59f931e 100644
--- a/Netgen/libsrc/interface/nglib.cpp
+++ b/Netgen/libsrc/interface/nglib.cpp
@@ -20,14 +20,32 @@
 #include <geometry2d.hpp>
 #include <meshing.hpp>
 
+
+
 // #include <FlexLexer.h>
 
+namespace netgen {
+  extern void MeshFromSpline2D (SplineGeometry2d & geometry,
+				Mesh *& mesh, 
+				MeshingParameters & mp);
+}
+
+
+
+
+
+
+
+namespace nglib {
 #include "nglib.h"
+}
 
 using namespace netgen;
 
 // constants and types:
 
+namespace nglib
+{
 // initialize, deconstruct Netgen library:
 void Ng_Init ()
 {
@@ -164,7 +182,7 @@ Ng_Result Ng_GenerateVolumeMesh (Ng_Mesh * mesh, Ng_Meshing_Parameters * mp)
 
   MeshVolume (mparam, *m);
   RemoveIllegalElements (*m);
-  OptimizeVolume (mparam, *m, NULL);
+  OptimizeVolume (mparam, *m);
 
   return NG_OK;
 }
@@ -242,27 +260,23 @@ void Ng_GetSegment_2D (Ng_Mesh * mesh, int num, int * pi, int * matnum)
 
 
 
-Ng_Geometry_2D * Ng_LoadGeometry_2D (char * filename)
+
+Ng_Geometry_2D * Ng_LoadGeometry_2D (const char * filename)
 {
   SplineGeometry2d * geom = new SplineGeometry2d();
   geom -> Load (filename);
   return (Ng_Geometry_2D *)geom;
 }
 
-namespace netgen {
-  extern void MeshFromSpline2D (SplineGeometry2d & geometry,
-				Mesh *& mesh, 
-				MeshingParameters & mp);
-}
-
-
 Ng_Result Ng_GenerateMesh_2D (Ng_Geometry_2D * geom,
 			      Ng_Mesh ** mesh,
 			      Ng_Meshing_Parameters * mp)
 {
-  MeshingParameters mparam;
+  // use global variable mparam
+  //  MeshingParameters mparam;  
   mparam.maxh = mp->maxh;
   mparam.meshsizefilename = mp->meshsize_filename;
+  mparam.quad = mp->quad_dominated;
 
   Mesh * m;
   MeshFromSpline2D (*(SplineGeometry2d*)geom, m, mparam);
@@ -270,10 +284,20 @@ Ng_Result Ng_GenerateMesh_2D (Ng_Geometry_2D * geom,
   cout << m->GetNSE() << " elements, " << m->GetNP() << " points" << endl;
   
   *mesh = (Ng_Mesh*)m;
+  return NG_OK;
 }
 
 
 
+void Ng_HP_Refinement (Ng_Geometry_2D * geom,
+		       Ng_Mesh * mesh,
+		       int levels)
+{
+  Refinement2d ref(*(SplineGeometry2d*)geom);
+  HPRefinement (*(Mesh*)mesh, &ref, levels);
+}
+
+
 
 
 
@@ -288,13 +312,20 @@ Ng_Result Ng_GenerateMesh_2D (Ng_Geometry_2D * geom,
 ARRAY<STLReadTriangle> readtrias; //only before initstlgeometry
 ARRAY<Point<3> > readedges; //only before init stlgeometry
  
-void Ng_SaveMesh(Ng_Mesh * mesh, char* filename)
+void Ng_SaveMesh(Ng_Mesh * mesh, const char* filename)
 {
   ((Mesh*)mesh)->Save(filename);
 }
 
+Ng_Mesh * Ng_LoadMesh(const char* filename)
+{
+  Mesh * mesh = new Mesh;
+  mesh->Load(filename);
+  return ( (Ng_Mesh*)mesh );
+}
+
 // loads geometry from STL file
-Ng_STL_Geometry * Ng_STL_LoadGeometry (char * filename, int binary)
+Ng_STL_Geometry * Ng_STL_LoadGeometry (const char * filename, int binary)
 {
   int i;
   STLGeometry geom;
@@ -387,12 +418,15 @@ Ng_Result Ng_STL_MakeEdges (Ng_STL_Geometry * geom,
 		   stlgeometry->GetBoundingBox().PMax() + Vec3d(10, 10, 10),
 		   0.3);
 
+  me -> LoadLocalMeshSize (mp->meshsize_filename);
+  /*
   if (mp->meshsize_filename)
     {
       ifstream infile (mp->meshsize_filename);
       if (!infile.good()) return NG_FILE_NOT_FOUND;
       me -> LoadLocalMeshSize (infile);
     }
+  */
 
   STLMeshing (*stlgeometry, *me);
   
@@ -496,10 +530,11 @@ Ng_Meshing_Parameters :: Ng_Meshing_Parameters()
   fineness = 0.5;
   secondorder = 0;
   meshsize_filename = 0;
+  quad_dominated = 0;
 }
 
 
-
+}
 
 
 // compatibility functions:
diff --git a/Netgen/libsrc/interface/nglib.h b/Netgen/libsrc/interface/nglib.h
index 592517c945..20d745d470 100644
--- a/Netgen/libsrc/interface/nglib.h
+++ b/Netgen/libsrc/interface/nglib.h
@@ -13,8 +13,6 @@
   
 */
 
-
-
 /// Data type for NETGEN mesh
 typedef void * Ng_Mesh;
 
@@ -34,7 +32,7 @@ typedef void * Ng_STL_Geometry;
 
 // implemented element types:
 enum Ng_Volume_Element_Type { NG_TET = 1, NG_PYRAMID = 2, NG_PRISM = 3,
-			      NG_TET10 = 4 };
+				 NG_TET10 = 4 };
 
 // max number of nodes per surface element
 #define NG_SURFACE_ELEMENT_MAXPOINTS 6
@@ -53,6 +51,7 @@ class Ng_Meshing_Parameters
   double fineness;   // 0 .. coarse, 1 .. fine
   int secondorder;
   char * meshsize_filename;
+  int quad_dominated;
 
   Ng_Meshing_Parameters();
 };
@@ -69,10 +68,10 @@ enum Ng_Result { NG_OK = 0,
 
 
 
-#ifdef __cplusplus
-extern "C" 
-{
-#endif
+// #ifdef __cplusplus
+// extern "C" 
+// {
+// #endif
   
   // initialize, deconstruct Netgen library:
   void Ng_Init ();
@@ -115,7 +114,8 @@ extern "C"
   // generates volume mesh from surface mesh
   Ng_Result Ng_GenerateVolumeMesh (Ng_Mesh * mesh, Ng_Meshing_Parameters * mp);
 
-  void Ng_SaveMesh(Ng_Mesh * mesh, char* filename);
+  void Ng_SaveMesh(Ng_Mesh * mesh, const char* filename);
+  Ng_Mesh * Ng_LoadMesh(const char* filename);
 
 
 
@@ -147,14 +147,16 @@ extern "C"
 
 
   // load 2d netgen spline geometry
-  Ng_Geometry_2D * Ng_LoadGeometry_2D (char * filename);
+  Ng_Geometry_2D * Ng_LoadGeometry_2D (const char * filename);
 
   // generate 2d mesh, mesh is allocated by function
   Ng_Result Ng_GenerateMesh_2D (Ng_Geometry_2D * geom,
 				Ng_Mesh ** mesh,
 				Ng_Meshing_Parameters * mp);
   
-
+  void Ng_HP_Refinement (Ng_Geometry_2D * geom,
+			 Ng_Mesh * mesh,
+			 int levels);
   
 
 
@@ -166,7 +168,7 @@ extern "C"
 
 
   // loads geometry from STL file
-  Ng_STL_Geometry * Ng_STL_LoadGeometry (char * filename, int binary = 0);
+  Ng_STL_Geometry * Ng_STL_LoadGeometry (const char * filename, int binary = 0);
 
 
   // generate new STL Geometry
@@ -198,9 +200,9 @@ extern "C"
 					Ng_Mesh * mesh,
 					Ng_Meshing_Parameters * mp);
 
-#ifdef __cplusplus
-  }
-#endif
+// #ifdef __cplusplus
+//  }
+// #endif
 
 
 #endif
diff --git a/Netgen/libsrc/interface/readuser.cpp b/Netgen/libsrc/interface/readuser.cpp
index 0a9d2b6802..f7d4d1eae8 100644
--- a/Netgen/libsrc/interface/readuser.cpp
+++ b/Netgen/libsrc/interface/readuser.cpp
@@ -16,10 +16,12 @@ namespace netgen
 #include "writeuser.hpp"
 
 void ReadFile (Mesh & mesh,
-	       const char * filename)
+	       const string & hfilename)
 {
   cout << "Read User File" << endl;
 
+  const char * filename = hfilename.c_str();
+
   int i, j;
 
   char reco[100];
@@ -70,7 +72,7 @@ void ReadFile (Mesh & mesh,
 	  
 	  /*
 	    if (invert)
-	    Swap (el.PNum(2), el.PNum(3));
+	    swap (el.PNum(2), el.PNum(3));
 	  */
 	  
 	  mesh.AddSurfaceElement (el);
@@ -154,7 +156,7 @@ void ReadFile (Mesh & mesh,
 		  in >> el.PNum(1) >> el.PNum(2) >> el.PNum(3);
 	      
 		  if (invert)
-		    Swap (el.PNum(2), el.PNum(3));
+		    swap (el.PNum(2), el.PNum(3));
 		  mesh.AddSurfaceElement (el);	  
 	      
 		  for (j = 1; j <= 5; j++)
@@ -200,7 +202,7 @@ void ReadFile (Mesh & mesh,
 	{
 	  int mat, nelp;
 	  in >> mat >> nelp;
-	  Element2d el (nelp);
+	  Element2d el (nelp == 3 ? TRIG : QUAD);
 	  el.SetIndex (mat);
 	  for (j = 1; j <= nelp; j++)
 	    in >> el.PNum(j);
@@ -222,42 +224,62 @@ void ReadFile (Mesh & mesh,
     {
       cout << "Reading Neutral Format" << endl;
       
-      char buf[100];
       int np, ne, nse, i, j;
 
       ifstream in (filename);
 
       in >> np;
-      for (i = 1; i <= np; i++)
-	{
-	  Point3d p(0,0,0);
-	  in >> p.X() >> p.Y() >> p.Z();
-	  mesh.AddPoint (p);
-	}
 
-      in >> ne;
-      for (i = 1; i <= ne; i++)
+      if (in.good())
 	{
-	  int mat;
-	  in >> mat;
-	  Element el (4);
-	  el.SetIndex (mat);
-	  for (j = 1; j <= 4; j++)
-	    in >> el.PNum(j);
-	  mesh.AddVolumeElement (el);
-	}
+	  // file starts with an integer
 
-
-      in >> nse;
-      for (i = 1; i <= nse; i++)
+	  for (i = 1; i <= np; i++)
+	    {
+	      Point3d p(0,0,0);
+	      in >> p.X() >> p.Y() >> p.Z();
+	      mesh.AddPoint (p);
+	    }
+	  
+	  in >> ne;
+	  for (i = 1; i <= ne; i++)
+	    {
+	      int mat;
+	      in >> mat;
+	      Element el (4);
+	      el.SetIndex (mat);
+	      for (j = 1; j <= 4; j++)
+		in >> el.PNum(j);
+	      mesh.AddVolumeElement (el);
+	    }
+	  
+	  in >> nse;
+	  for (i = 1; i <= nse; i++)
+	    {
+	      int mat, nelp;
+	      in >> mat;
+	      Element2d el (TRIG);
+	      el.SetIndex (mat);
+	      for (j = 1; j <= 3; j++)
+		in >> el.PNum(j);
+	      mesh.AddSurfaceElement (el);
+	    }
+	}
+      else
 	{
-	  int mat, nelp;
-	  in >> mat;
-	  Element2d el (3);
-	  el.SetIndex (mat);
-	  for (j = 1; j <= 3; j++)
-	    in >> el.PNum(j);
-	  mesh.AddSurfaceElement (el);
+	  char buf[100];
+	  in.clear();
+	  do
+	    {
+	      in >> buf;
+	      cout << "buf = " << buf << endl;
+	      if (strcmp (buf, "points") == 0)
+		{
+		  in >> np;
+		  cout << "np = " << np << endl;
+		}
+	    }
+	  while (in.good());
 	}
     }
 
@@ -306,7 +328,7 @@ void ReadFile (Mesh & mesh,
 	{
 	  inemt >> p1 >> p2 >> p3 >> bcprop >> value;
 
-	  if (bcprop < 1 >> bcprop > 4)
+	  if (bcprop < 1 || bcprop > 4)
 	    cerr << "bcprop out of range, bcprop = " << bcprop << endl;
 	  p1++;
 	  p2++;
diff --git a/Netgen/libsrc/interface/writeelmer.cpp b/Netgen/libsrc/interface/writeelmer.cpp
new file mode 100644
index 0000000000..caf02e2c68
--- /dev/null
+++ b/Netgen/libsrc/interface/writeelmer.cpp
@@ -0,0 +1,127 @@
+
+//
+//  Write Elmer file
+//
+//
+
+#include <mystdlib.h>
+
+#include <myadt.hpp>
+#include <linalg.hpp>
+#include <csg.hpp>
+#include <meshing.hpp>
+
+namespace netgen
+{
+#include "writeuser.hpp"
+
+#include <sys/stat.h>
+
+
+void WriteElmerFormat (const Mesh &mesh,
+			 const string &filename)
+{
+  cout << "write elmer mesh files" << endl;
+  int np = mesh.GetNP();
+  int ne = mesh.GetNE();
+  int nse = mesh.GetNSE();
+  int i, j, rc;
+  char str[200];
+  
+  int inverttets = mparam.inverttets;
+  int invertsurf = mparam.inverttrigs;
+
+#ifdef WIN32
+  cerr << "not yet implemented for Windows platforms." << endl;
+  return;
+#else
+  rc = mkdir(filename.c_str(), S_IRWXU|S_IRWXG);
+#endif
+  sprintf( str, "%s/mesh.header", filename.c_str() );
+  ofstream outfile_h(str);
+  sprintf( str, "%s/mesh.nodes", filename.c_str() );
+  ofstream outfile_n(str);
+  sprintf( str, "%s/mesh.elements", filename.c_str() );
+  ofstream outfile_e(str);
+  sprintf( str, "%s/mesh.boundary", filename.c_str() );
+  ofstream outfile_b(str);
+
+  // fill hashtable
+
+  INDEX_3_HASHTABLE<int> face2volelement(ne);
+
+  for (i = 1; i <= ne; i++)
+    {
+      const Element & el = mesh.VolumeElement(i);
+      INDEX_3 i3;
+      int k, l;
+      for (j = 1; j <= 4; j++)   // loop over faces of tet
+	{
+	  l = 0;
+	  for (k = 1; k <= 4; k++)
+	    if (k != j)
+	      {
+		l++;
+		i3.I(l) = el.PNum(k);
+	      }
+	  i3.Sort();
+	  face2volelement.Set (i3, i);
+	}
+    }
+
+//  outfile.precision(6);
+//  outfile.setf (ios::fixed, ios::floatfield);
+//  outfile.setf (ios::showpoint);
+  
+  outfile_h << np << " " << ne << " " << nse << "\n";
+  outfile_h << "1"     << "\n";
+  outfile_h << "504 1" << "\n";
+  
+  for (i = 1; i <= np; i++)
+    {
+      const Point3d & p = mesh.Point(i);
+      
+      outfile_n << i << " -1 ";
+      outfile_n << p.X() << " ";
+      outfile_n << p.Y() << " ";
+      outfile_n << p.Z() << "\n";
+    }
+
+  for (i = 1; i <= ne; i++)
+    {
+      Element el = mesh.VolumeElement(i);
+      if (inverttets) el.Invert();
+      sprintf( str, "5%02d", (int)el.GetNP() );
+      outfile_e << i << " " << el.GetIndex() << " " << str <<  "  ";
+      for (j = 1; j <= el.GetNP(); j++)
+	{
+	  outfile_e << " ";
+	  outfile_e << el.PNum(j);
+	}
+      outfile_e << "\n";
+    }
+
+  for (i = 1; i <= nse; i++)
+    {
+      Element2d el = mesh.SurfaceElement(i);
+      if (invertsurf) el.Invert();
+      sprintf( str, "3%02d", (int)el.GetNP() );
+      {
+	  INDEX_3 i3;
+	  for (j = 1; j <= 3; j++) i3.I(j) = el.PNum(j);
+	  i3.Sort();
+	  
+	  int elind = face2volelement.Get(i3);
+          outfile_b << i << " " << mesh.GetFaceDescriptor(el.GetIndex()).BCProperty() << 
+         " " << elind << " 0 "  << str << "    ";
+      }
+      for (j = 1; j <= el.GetNP(); j++)
+	{
+	  outfile_b << " ";
+	  outfile_b << el.PNum(j);
+	}
+      outfile_b << "\n";
+    }
+}
+
+}
diff --git a/Netgen/libsrc/interface/writefluent.cpp b/Netgen/libsrc/interface/writefluent.cpp
index 1a214089e3..5c08d59857 100644
--- a/Netgen/libsrc/interface/writefluent.cpp
+++ b/Netgen/libsrc/interface/writefluent.cpp
@@ -17,21 +17,10 @@ namespace netgen
 
 
 
-static char* CHex(int i)
-{
-  static char str[100];
-  sprintf(str,"%x",i); //hex
-  return str;
-}
-
-
-
-
 void WriteFluentFormat (const Mesh & mesh,
 			const string & filename)
 
 {
-  // Simple output 
   cout << "start writing fluent export" << endl;
       
   int np = mesh.GetNP();
@@ -46,7 +35,7 @@ void WriteFluentFormat (const Mesh & mesh,
   //outfile.setf (ios::fixed, ios::floatfield);
   //outfile.setf (ios::showpoint);
       
-  outfile << "(0 \"Exported file from NETGEN (c) Joachim Schoeberl and Johannes Gerstmayr\")" << endl;
+  outfile << "(0 \"Exported file from NETGEN \")" << endl;
   outfile << "(0 \"Dimension:\")" << endl;
   outfile << "(2 3)" << endl << endl;
 
@@ -129,7 +118,6 @@ void WriteFluentFormat (const Mesh & mesh,
 	  int eli2 = 0;
 	  int stopsig = 0;
 	      
-
 	  for (i2 = 1; i2 <= nel; i2++)
 	    {
 	      locind = locels.Get(i2);
@@ -153,8 +141,11 @@ void WriteFluentFormat (const Mesh & mesh,
 	  if (eli2 > i) //dont write faces two times!
 	    {
 	      //i: left cell, eli: right cell
-	      outfile << CHex(face.PNum(2)) << " " << CHex(face.PNum(1)) << " " << CHex(face.PNum(3)) << " ";
-	      outfile << CHex(i) << " " << CHex(eli2) << "\n";
+	      outfile << hex << face.PNum(2) << " "
+		<< hex << face.PNum(1) << " "
+		<< hex << face.PNum(3) << " "
+		<< hex << i  << " "
+		<< hex << eli2 << "\n";
 	    }
 	  if (eli2 == 0) 
 	    {
@@ -170,8 +161,10 @@ void WriteFluentFormat (const Mesh & mesh,
 
   for (i = 1; i <= surfaceelp.Size(); i++)
     {
-      outfile << CHex(surfaceelp.Get(i).I1()) << " " << CHex(surfaceelp.Get(i).I2()) << " " << CHex(surfaceelp.Get(i).I3()) << " ";
-      outfile << CHex(surfaceeli.Get(i)) << " " << 0 << "\n";
+      outfile << hex << surfaceelp.Get(i).I1() << " "
+	      << hex << surfaceelp.Get(i).I2() << " "
+	      << hex << surfaceelp.Get(i).I3() << " "
+	      << hex << surfaceeli.Get(i) << " " << 0 << "\n";
     }
 
   outfile << "))" << endl << endl;
diff --git a/Netgen/libsrc/interface/writegmsh.cpp b/Netgen/libsrc/interface/writegmsh.cpp
new file mode 100644
index 0000000000..93def67783
--- /dev/null
+++ b/Netgen/libsrc/interface/writegmsh.cpp
@@ -0,0 +1,200 @@
+/*************************************
+ * Write Gmsh file
+ * First issue the 04/26/2004 by Paul CARRICO (paul.carrico@free.fr)
+ * At the moment, the GMSH format is available for
+ * linear tetrahedron elements i.e. in 3D
+ * (based on Neutral Format)
+ *
+ * Second issue the 05/05/2004 by Paul CARRICO
+ * Thanks to Joachim Schoeberl for the correction of a minor bug
+ * the 2 initial Gmsh Format (i.e. volume format and surface format) are group together)
+ * in only one file
+ **************************************/
+
+#include <mystdlib.h>
+
+#include <myadt.hpp>
+#include <linalg.hpp>
+#include <csg.hpp>
+#include <meshing.hpp>
+
+namespace netgen
+{
+#include "writeuser.hpp"
+
+
+
+/*
+ *  GMSH mesh format
+ *  points, elements, surface elements and physical entities
+ */
+
+void WriteGmshFormat (const Mesh & mesh,
+			 const CSGeometry & geom,
+			 const string & filename)
+{
+  ofstream outfile (filename.c_str());
+  outfile.precision(6);
+  outfile.setf (ios::fixed, ios::floatfield);
+  outfile.setf (ios::showpoint);
+
+  int np = mesh.GetNP();  /// number of point
+  int ne = mesh.GetNE();  /// number of element
+  int nse = mesh.GetNSE();  /// number of surface element (BC)
+  int i, j, k, l;
+
+
+  /*
+   * 3D section : Linear volume elements (only tetrahedra)
+   */
+
+   if (ne > 0 && mesh.VolumeElement(1).GetNP() == 4)
+      {
+      cout << "Write GMSH Format \n";
+      cout << "The GMSH format is available for linear tetrahedron elements only in 3D\n" << endl;
+
+      int inverttets = mparam.inverttets;
+      int invertsurf = mparam.inverttrigs;
+
+
+      /// Write nodes
+      outfile << "$NOD\n";
+      outfile << np << "\n";
+  
+      for (i = 1; i <= np; i++)
+          {
+          const Point3d & p = mesh.Point(i);
+          outfile << i << " "; /// node number
+          outfile << p.X() << " ";
+          outfile << p.Y() << " ";
+          outfile << p.Z() << "\n";
+          }
+      outfile << "$ENDNOD\n";
+
+      /// write elements
+      outfile << "$ELM\n";
+      outfile << ne + nse << "\n";  ////  number of elements + number of surfaces BC
+
+     for (i = 1; i <= nse; i++)
+         {
+         Element2d el = mesh.SurfaceElement(i);
+         if (invertsurf) el.Invert();
+         outfile << i;
+         outfile << " ";
+         outfile << "2";
+         outfile << " ";
+         outfile << mesh.GetFaceDescriptor (el.GetIndex()).BCProperty() << " ";
+         /// that means that physical entity = elementary entity (arbitrary approach)
+         outfile << mesh.GetFaceDescriptor (el.GetIndex()).BCProperty() << " ";
+         outfile << "3";
+         outfile << " ";
+                 for (j = 1; j <= el.GetNP(); j++)
+                     {
+                     outfile << " ";
+                     outfile << el.PNum(j);
+                     }
+                     outfile << "\n";
+         }
+
+
+         for (i = 1; i <= ne; i++)
+             {
+             Element el = mesh.VolumeElement(i);
+             if (inverttets) el.Invert();
+             outfile << nse + i; /// element number
+             outfile << " ";
+             outfile << "4"; /// element type i.e. Tetraedron == 4
+             outfile << " ";
+             outfile << 100000 + el.GetIndex();
+             /// that means that physical entity = elementary entity (arbitrary approach)
+             outfile << " ";
+             outfile << 100000 + el.GetIndex();   /// volume number
+             outfile << " ";
+             outfile << "4";  /// number of nodes i.e. 4 for a tetrahedron
+                                                                                                        
+             for (j = 1; j <= el.GetNP(); j++)
+                 {
+                 outfile << " ";
+                 outfile << el.PNum(j);
+                 }
+             outfile << "\n";
+             }
+
+
+             outfile << "$ENDELM\n";
+   }
+
+   /*
+    * End of 3D section
+    */
+
+
+
+
+
+  /*
+   * 2D section : available for triangles and quadrangles
+   */
+      else if (ne == 0)   /// means that there's no 3D element
+              {
+              cout << "\n Write Gmsh Surface Mesh (triangle and/or quadrangles)" << endl;
+
+              /// Write nodes
+              outfile << "$NOD\n";
+              outfile << np << "\n";
+
+              for (i = 1; i <= np; i++)
+              {
+              const Point3d & p = mesh.Point(i);
+              outfile << i << " "; /// node number
+              outfile << p.X() << " ";
+              outfile << p.Y() << " ";
+              outfile << p.Z() << "\n";
+              }
+              outfile << "$ENDNOD\n";
+
+
+              /// write triangles & quadrangles
+              outfile << "$ELM\n";
+              outfile << nse << "\n";
+
+              for (k = 1; k <= nse; k++)
+              {
+              const Element2d & el = mesh.SurfaceElement(k);
+
+
+              outfile << k;
+              outfile << " ";
+              outfile << (el.GetNP()-1);   // 2 for a triangle and 3 for a quadrangle
+              outfile << " ";
+              outfile << mesh.GetFaceDescriptor (el.GetIndex()).BCProperty() << " ";
+              /// that means that physical entity = elementary entity (arbitrary approach)
+              outfile << mesh.GetFaceDescriptor (el.GetIndex()).BCProperty() << " ";
+              outfile << (el.GetNP());    // number of node per surfacic element
+              outfile << " ";
+
+              for (l = 1; l <= el.GetNP(); l++)
+                  {
+                  outfile << " ";
+                  outfile << el.PNum(l);
+                  }
+	              outfile << "\n";
+		  
+               }
+               outfile << "$ENDELM$ \n";
+    }
+
+   /*
+    * End of 2D section
+    */
+
+     else
+    {
+    cout << " Invalide element type for Gmsh volume Format !\n";
+    }
+
+
+}
+}
+
+
diff --git a/Netgen/libsrc/interface/writepermas.cpp b/Netgen/libsrc/interface/writepermas.cpp
index 824a36bc9b..9dc7662458 100644
--- a/Netgen/libsrc/interface/writepermas.cpp
+++ b/Netgen/libsrc/interface/writepermas.cpp
@@ -10,160 +10,199 @@
 #include <csg.hpp>
 #include <meshing.hpp>
 
-namespace netgen
-{
-#include "writeuser.hpp"
+#include <string>
 
+using namespace std;
 
-
-void WritePermasFormat (const Mesh & mesh,
-			const string & filename)
-
+namespace netgen
 {
-  
-  ofstream outfile (filename.c_str());
-
-  outfile.precision(8);
-      
-  int np = mesh.GetNP();
-  int ne = mesh.GetNE();
-  int nse = mesh.GetNSE();
-  int i, j, k;
+#include "writeuser.hpp"
+    // Forward declarations (don't know, where to define them, sorry)
+    int addComponent(string &strComp, string &strSitu, ofstream &out);
 
 
-  if (ne == 0)
+    // This should be the new function to export a PERMAS file
+    void WritePermasFormat (const Mesh &mesh, const string &filename, 
+                            string &strComp, string &strSitu) 
     {
-      // pure surface mesh
-
-      cout << "\nWrite Permas Surface Mesh" << endl;
-
-      int elnr = 0;
-      for (j = 1; j <= 2; j++)
-	{
-	  int nelp;
-	  switch (j)
-	    {
-	    case 1:
-	      nelp = 3;
-	      outfile << "$ELEMENT TYPE = TRIA3  ESET = ALLQUAD" << endl;		  
-	      break;
-	    case 2:
-	      nelp = 4;
-	      outfile << "$ELEMENT TYPE = QUAD4  ESET = ALLQUAD" << endl;		  
-	      break;
-	    }
-	      
-	  for (i = 1; i <= nse; i++)
-	    {
-	      const Element2d & el = mesh.SurfaceElement(i);
-	      if (el.GetNP() != nelp)
-		continue;
-		  
-	      elnr++;
-	      outfile << elnr << "  ";
-	      for (k = 1; k <= nelp; k++)
-		outfile << " " << el.PNum(k);
-	      outfile << endl;
-		  
-	    }
-	}
+        ofstream outfile (filename.c_str());
+        addComponent(strComp, strSitu, outfile);
+        WritePermasFormat ( mesh, filename);
     }
 
-  else
-
+    void WritePermasFormat (const Mesh &mesh, const string &filename)
     {
-      cout << "\nWrite Permas Volume Mesh" << endl;
-
-
-      int secondorder = (mesh.VolumeElement(1).GetNP() == 10);
-	  	  
-      if (!secondorder)
-	{
-	  outfile << "$ELEMENT TYPE = TET4  ESET = ALLTET" << endl;
-	  for (i = 1; i <= ne; i++)
-	    {
-	      const Element & el = mesh.VolumeElement(i);
-	      outfile << i 
-		      << " " << el.PNum(1) 
-		      << " " << el.PNum(2) 
-		      << " " << el.PNum(3) 
-		      << " " << el.PNum(4) << endl;
-	    }
-	}
-      else
-	{
-	  outfile << "$ELEMENT TYPE = TET10  ESET = ALLTET" << endl;
-	  for (i = 1; i <= ne; i++)
-	    {
-	      const Element & el = mesh.VolumeElement(i);
-	      outfile << i 
-		      << " " << el.PNum(1) 
-		      << " " << el.PNum(5) 
-		      << " " << el.PNum(2) 
-		      << " " << el.PNum(8) 
-		      << " " << el.PNum(3) 
-		      << " " << el.PNum(6) << endl << "& "
-		      << " " << el.PNum(7) 
-		      << " " << el.PNum(9) 
-		      << " " << el.PNum(10) 
-		      << " " << el.PNum(4) << endl;
-	    }
-	}
-	  
-      outfile << endl << endl;
-	  
-	  
-      outfile << "$SURFACE GEO  SURFID = 1  SFSET = ALLSUR" << endl;
-      for (i = 1; i <= nse; i++)
-	{
-	  const Element2d & el = mesh.SurfaceElement(i);
-	  if (el.GetNP() == 3)
-	    outfile << "STRIA3"
-		    << " " << el.PNum(1) 
-		    << " " << el.PNum(2) 
-		    << " " << el.PNum(3) << endl;
-	}    
-	  
-      for (i = 1; i <= nse; i++)
-	{
-	  const Element2d & el = mesh.SurfaceElement(i);
-	  if (el.GetNP() == 4)
-	    outfile << "SQUAD4"
-		    << " " << el.PNum(1) 
-		    << " " << el.PNum(2) 
-		    << " " << el.PNum(3) 
-		    << " " << el.PNum(4) << endl;
-	}      
-	  
-      for (i = 1; i <= nse; i++)
-	{
-	  const Element2d & el = mesh.SurfaceElement(i);
-	  if (el.GetNP() == 6)
-	    outfile << "STRIA6"
-		    << " " << el.PNum(1) 
-		    << " " << el.PNum(4) 
-		    << " " << el.PNum(2) 
-		    << " " << el.PNum(5) 
-		    << " " << el.PNum(3) 
-		    << " " << el.PNum(6) << endl;
-	}      
+        string strComp, strSitu;
+        ofstream outfile (filename.c_str());
+        
+        outfile.precision(8);
+
+        strSitu = strComp = "";
+        if (addComponent(strComp, strSitu, outfile) == 1) {
+            printf("Error while exporting PERMAS dat!\n");
+            return;
+        }
+    
+        int np = mesh.GetNP();
+        int ne = mesh.GetNE();
+        int nse = mesh.GetNSE();
+        int i, j, k;
+    
+        if (ne == 0)
+        {
+            // pure surface mesh
+            cout << "\nWrite Permas Surface Mesh" << endl;
+            
+            int elnr = 0;
+            for (j = 1; j <= 2; j++)
+            {
+                int nelp;
+                switch (j)
+                {
+                case 1:
+                    nelp = 3;
+                    outfile << "$ELEMENT TYPE = TRIA3  ESET = ALLQUAD" << endl;		  
+                    break;
+                case 2:
+                    nelp = 4;
+                    outfile << "$ELEMENT TYPE = QUAD4  ESET = ALLQUAD" << endl;		  
+                    break;
+                }
+                
+                for (i = 1; i <= nse; i++)
+                {
+                    const Element2d & el = mesh.SurfaceElement(i);
+                    if (el.GetNP() != nelp)
+                        continue;
+                    
+                    elnr++;
+                    outfile << elnr << "  ";
+                    for (k = 1; k <= nelp; k++)
+                        outfile << " " << el.PNum(k);
+                    outfile << endl;
+                    
+                }
+            }
+        }
+        else
+        {
+            cout << "\nWrite Permas Volume Mesh" << endl;
+            
+            int secondorder = (mesh.VolumeElement(1).GetNP() == 10);
+            
+            if (!secondorder)
+            {
+                outfile << "$ELEMENT TYPE = TET4  ESET = ALLTET" << endl;
+                for (i = 1; i <= ne; i++)
+                {
+                    const Element & el = mesh.VolumeElement(i);
+                    outfile << i 
+                            << " " << el.PNum(1) 
+                            << " " << el.PNum(2) 
+                            << " " << el.PNum(3) 
+                            << " " << el.PNum(4) << endl;
+                }
+            }
+            else
+            {
+                outfile << "$ELEMENT TYPE = TET10  ESET = ALLTET" << endl;
+                for (i = 1; i <= ne; i++)
+                {
+                    const Element & el = mesh.VolumeElement(i);
+                    outfile << i 
+                            << " " << el.PNum(1) 
+                            << " " << el.PNum(5) 
+                            << " " << el.PNum(2) 
+                            << " " << el.PNum(8) 
+                            << " " << el.PNum(3) 
+                            << " " << el.PNum(6) << endl << "& "
+                            << " " << el.PNum(7) 
+                            << " " << el.PNum(9) 
+                            << " " << el.PNum(10) 
+                            << " " << el.PNum(4) << endl;
+                }
+            }
+            
+            outfile << endl << endl;
+            
+            
+            outfile << "$SURFACE GEO  SURFID = 1  SFSET = ALLSUR" << endl;
+            for (i = 1; i <= nse; i++)
+            {
+                const Element2d & el = mesh.SurfaceElement(i);
+                if (el.GetNP() == 3)
+                    outfile << "STRIA3"
+                            << " " << el.PNum(1) 
+                            << " " << el.PNum(2) 
+                            << " " << el.PNum(3) << endl;
+            }    
+            
+            for (i = 1; i <= nse; i++)
+            {
+                const Element2d & el = mesh.SurfaceElement(i);
+                if (el.GetNP() == 4)
+                    outfile << "SQUAD4"
+                            << " " << el.PNum(1) 
+                            << " " << el.PNum(2) 
+                            << " " << el.PNum(3) 
+                            << " " << el.PNum(4) << endl;
+            }      
+            
+            for (i = 1; i <= nse; i++)
+            {
+                const Element2d & el = mesh.SurfaceElement(i);
+                if (el.GetNP() == 6)
+                    outfile << "STRIA6"
+                            << " " << el.PNum(1) 
+                            << " " << el.PNum(4) 
+                            << " " << el.PNum(2) 
+                            << " " << el.PNum(5) 
+                            << " " << el.PNum(3) 
+                            << " " << el.PNum(6) << endl;
+            }      
+        }
+        
+        
+        outfile << endl << endl;
+        
+        outfile << "$COOR  NSET = ALLNODES" << endl;
+        
+        outfile.precision(6);
+        outfile.setf (ios::fixed, ios::floatfield);
+        outfile.setf (ios::showpoint);
+        
+        for (i = 1; i <= np; i++)
+        {
+            outfile << i << " ";
+            outfile << mesh.Point(i).X() << " ";
+            outfile << mesh.Point(i).Y() << " ";
+            outfile << mesh.Point(i).Z() << "\n";
+        }
     }
-
-
-  outfile << endl << endl;
-      
-
-
-  outfile << "$COOR  NSET = ALLNODES" << endl;
-
-  for (i = 1; i <= np; i++)
+    ////////////////////////////////////////////////////////////////////////////////// 
+    // \brief Writes PERMAS configuration header into export file
+    //        Returns >0 in case of errors
+    // \par string &strComp  : Reference to component description
+    // \par string &strComp  : Reference to situation description
+    ////////////////////////////////////////////////////////////////////////////////// 
+    int addComponent(string &strComp, string &strSitu, ofstream &out)
     {
-      outfile << i << " ";
-      outfile << mesh.Point(i).X() << " ";
-      outfile << mesh.Point(i).Y() << " ";
-      outfile << mesh.Point(i).Z() << "\n";
+        if (strComp.size() > 12 || strSitu > 12) 
+            return 1;
+
+        if (0 == strComp.size()) 
+            strComp = "KOMPO1";
+        
+        if (0 == strSitu.size())
+            strSitu = "SIT1";
+
+        // Writing description header of configuration
+        out << "$ENTER COMPONENT  NAME = " << strComp << "  DOFTYPE = DISP MATH" << endl << endl;
+        out << "   $SITUATION  NAME = " << strSitu << endl;
+        out << "   $END SITUATION" << endl << endl;
+        out << "   $STRUCTURE" << endl;
+        
+        return 0;
     }
-}
-
-
+    
 }
diff --git a/Netgen/libsrc/interface/writepermas2.cpp b/Netgen/libsrc/interface/writepermas2.cpp
new file mode 100644
index 0000000000..f78714c5a4
--- /dev/null
+++ b/Netgen/libsrc/interface/writepermas2.cpp
@@ -0,0 +1,173 @@
+//
+// Write Permas file
+// for Intes GmbH, Stuttgart
+//
+
+#include <mystdlib.h>
+
+#include <myadt.hpp>
+#include <linalg.hpp>
+#include <csg.hpp>
+#include <meshing.hpp>
+
+namespace netgen
+{
+#include "writeuser.hpp"
+
+
+
+void WritePermasFormat (const Mesh & mesh,
+			const string & filename)
+
+{
+  
+  ofstream outfile (filename.c_str());
+
+  outfile.precision(8);
+      
+  int np = mesh.GetNP();
+  int ne = mesh.GetNE();
+  int nse = mesh.GetNSE();
+  int i, j, k;
+
+
+  if (ne == 0)
+    {
+      // pure surface mesh
+
+      cout << "\nWrite Permas Surface Mesh" << endl;
+
+      int elnr = 0;
+      for (j = 1; j <= 2; j++)
+	{
+	  int nelp;
+	  switch (j)
+	    {
+	    case 1:
+	      nelp = 3;
+	      outfile << "$ELEMENT TYPE = TRIA3  ESET = ALLQUAD" << endl;		  
+	      break;
+	    case 2:
+	      nelp = 4;
+	      outfile << "$ELEMENT TYPE = QUAD4  ESET = ALLQUAD" << endl;		  
+	      break;
+	    }
+	      
+	  for (i = 1; i <= nse; i++)
+	    {
+	      const Element2d & el = mesh.SurfaceElement(i);
+	      if (el.GetNP() != nelp)
+		continue;
+		  
+	      elnr++;
+	      outfile << elnr << "  ";
+	      for (k = 1; k <= nelp; k++)
+		outfile << " " << el.PNum(k);
+	      outfile << endl;
+		  
+	    }
+	}
+    }
+
+  else
+
+    {
+      cout << "\nWrite Permas Volume Mesh" << endl;
+
+
+      int secondorder = (mesh.VolumeElement(1).GetNP() == 10);
+	  	  
+      if (!secondorder)
+	{
+	  outfile << "$ELEMENT TYPE = TET4  ESET = ALLTET" << endl;
+	  for (i = 1; i <= ne; i++)
+	    {
+	      const Element & el = mesh.VolumeElement(i);
+	      outfile << i 
+		      << " " << el.PNum(1) 
+		      << " " << el.PNum(2) 
+		      << " " << el.PNum(3) 
+		      << " " << el.PNum(4) << endl;
+	    }
+	}
+      else
+	{
+	  outfile << "$ELEMENT TYPE = TET10  ESET = ALLTET" << endl;
+	  for (i = 1; i <= ne; i++)
+	    {
+	      const Element & el = mesh.VolumeElement(i);
+	      outfile << i 
+		      << " " << el.PNum(1) 
+		      << " " << el.PNum(5) 
+		      << " " << el.PNum(2) 
+		      << " " << el.PNum(8) 
+		      << " " << el.PNum(3) 
+		      << " " << el.PNum(6) << endl << "& "
+		      << " " << el.PNum(7) 
+		      << " " << el.PNum(9) 
+		      << " " << el.PNum(10) 
+		      << " " << el.PNum(4) << endl;
+	    }
+	}
+	  
+      outfile << endl << endl;
+	  
+	  
+      outfile << "$SURFACE GEO  SURFID = 1  SFSET = ALLSUR" << endl;
+      for (i = 1; i <= nse; i++)
+	{
+	  const Element2d & el = mesh.SurfaceElement(i);
+	  if (el.GetNP() == 3)
+	    outfile << "STRIA3"
+		    << " " << el.PNum(1) 
+		    << " " << el.PNum(2) 
+		    << " " << el.PNum(3) << endl;
+	}    
+	  
+      for (i = 1; i <= nse; i++)
+	{
+	  const Element2d & el = mesh.SurfaceElement(i);
+	  if (el.GetNP() == 4)
+	    outfile << "SQUAD4"
+		    << " " << el.PNum(1) 
+		    << " " << el.PNum(2) 
+		    << " " << el.PNum(3) 
+		    << " " << el.PNum(4) << endl;
+	}      
+	  
+      for (i = 1; i <= nse; i++)
+	{
+	  const Element2d & el = mesh.SurfaceElement(i);
+	  if (el.GetNP() == 6)
+	    outfile << "STRIA6"
+		    << " " << el.PNum(1) 
+		    << " " << el.PNum(4) 
+		    << " " << el.PNum(2) 
+		    << " " << el.PNum(5) 
+		    << " " << el.PNum(3) 
+		    << " " << el.PNum(6) << endl;
+	}      
+    }
+
+
+  outfile << endl << endl;
+      
+
+
+  outfile << "$COOR  NSET = ALLNODES" << endl;
+
+  outfile.precision(6);
+  outfile.setf (ios::fixed, ios::floatfield);
+  outfile.setf (ios::showpoint);
+
+  for (i = 1; i <= np; i++)
+    {
+      outfile << i << " ";
+      outfile << mesh.Point(i).X() << " ";
+      outfile << mesh.Point(i).Y() << " ";
+      outfile << mesh.Point(i).Z() << "\n";
+    }
+}
+
+
+}
diff --git a/Netgen/libsrc/interface/writetecplot.cpp b/Netgen/libsrc/interface/writetecplot.cpp
index 1f5e01351b..a3ddda3f9a 100644
--- a/Netgen/libsrc/interface/writetecplot.cpp
+++ b/Netgen/libsrc/interface/writetecplot.cpp
@@ -85,7 +85,7 @@ void WriteTecPlotFormat (const Mesh & mesh,
       for (i = 1; i <= np; i++)
 	if (sn.Elem(i) != 0)
 	  {
-	    geom.GetSurface(j) -> GetNormalVector ( mesh.Point(i), n );
+	    n = geom.GetSurface(j) -> GetNormalVector ( mesh.Point(i) );
 		
 	    outfile << mesh.Point(i).X() << " " /* Knoten Koordinaten */
 		    << mesh.Point(i).Y() << " "
diff --git a/Netgen/libsrc/interface/writeuser.cpp b/Netgen/libsrc/interface/writeuser.cpp
index 1909eb2193..83b9a050ce 100644
--- a/Netgen/libsrc/interface/writeuser.cpp
+++ b/Netgen/libsrc/interface/writeuser.cpp
@@ -28,9 +28,10 @@ void RegisterUserFormats (ARRAY<const char*> & names)
       "Fluent Format",
       "Permas Format",
       "FEAP Format",
+      "Elmer Format",
       "STL Format",
       "VRML Format",
-      "Fepp Format",
+      "Gmsh Format",
       //      { "Chemnitz Format" },
       0
     };
@@ -46,7 +47,8 @@ bool WriteUserFormat (const string & format,
 		      const CSGeometry & geom,
 		      const string & filename)
 {
-  cout << "Write user format " << format << endl;
+  PrintMessage (1, "Export mesh to file ", filename, 
+		", format is ", format);
 
   if (format == "Neutral Format")
     WriteNeutralFormat (mesh, geom, filename);
@@ -76,6 +78,9 @@ bool WriteUserFormat (const string & format,
   else if (format == "FEAP Format")
     WriteFEAPFormat (mesh, filename);
 
+  else if (format == "Elmer Format")
+    WriteElmerFormat (mesh, filename);
+
   else if (format == "STL Format")
     WriteSTLFormat (mesh, filename);
 
@@ -85,9 +90,15 @@ bool WriteUserFormat (const string & format,
   else if (format == "Fepp Format")
     WriteFEPPFormat (mesh, geom, filename);
 
+  else if (format ==  "EdgeElement Format")
+    WriteEdgeElementFormat (mesh, geom, filename);
+
   else if (format == "Chemnitz Format")
     WriteUserChemnitz (mesh, filename);
-  
+
+  else if (format == "Gmsh Format")
+    WriteGmshFormat (mesh, geom, filename);
+ 
   else 
     {
       return 1;
@@ -112,6 +123,7 @@ void WriteNeutralFormat (const Mesh & mesh,
   int np = mesh.GetNP();
   int ne = mesh.GetNE();
   int nse = mesh.GetNSE();
+  int nseg = mesh.GetNSeg();
   int i, j;
   
   int inverttets = mparam.inverttets;
@@ -133,25 +145,32 @@ void WriteNeutralFormat (const Mesh & mesh,
       outfile << p.X() << " ";
       outfile.width(9);
       outfile << p.Y() << " ";
-      outfile.width(9);
-      outfile << p.Z() << "\n";
+      if (mesh.GetDimension() == 3)
+	{
+	  outfile.width(9);
+	  outfile << p.Z();
+	  }
+      outfile << "\n";
     }
 
-  outfile << ne << "\n";
-  for (i = 1; i <= ne; i++)
+  if (mesh.GetDimension() == 3)
     {
-      Element el = mesh.VolumeElement(i);
-      if (inverttets)
-	el.Invert();
-      outfile.width(4);
-      outfile << el.GetIndex() << "  ";
-      for (j = 1; j <= el.GetNP(); j++)
+      outfile << ne << "\n";
+      for (i = 1; i <= ne; i++)
 	{
-	  outfile << " ";
-	  outfile.width(8);
-	  outfile << el.PNum(j);
+	  Element el = mesh.VolumeElement(i);
+	  if (inverttets)
+	    el.Invert();
+	  outfile.width(4);
+	  outfile << el.GetIndex() << "  ";
+	  for (j = 1; j <= el.GetNP(); j++)
+	    {
+	      outfile << " ";
+	      outfile.width(8);
+	      outfile << el.PNum(j);
+	    }
+	  outfile << "\n";
 	}
-      outfile << "\n";
     }
 
   outfile << nse << "\n";
@@ -170,6 +189,27 @@ void WriteNeutralFormat (const Mesh & mesh,
 	}
       outfile << "\n";
     }
+
+
+  if (mesh.GetDimension() == 2)
+    {
+      outfile << nseg << "\n";
+      for (i = 1; i <= nseg; i++)
+	{
+	  const Segment & seg = mesh.LineSegment(i);
+	  outfile.width(4);
+	  outfile << seg.si << "    ";
+
+	  outfile << " ";
+	  outfile.width(8);
+	  outfile << seg.p1;
+	  outfile << " ";
+	  outfile.width(8);
+	  outfile << seg.p2;
+
+	  outfile << "\n";
+	}
+    }
 }
 
 
@@ -561,6 +601,136 @@ void WriteFEPPFormat (const Mesh & mesh,
 
 
 
+/*
+ *  Edge element mesh format
+ *  points, elements, edges
+ */
+
+void WriteEdgeElementFormat (const Mesh & mesh,
+			     const CSGeometry & geom,
+			     const string & filename)
+{
+  cout << "write edge element format" << endl;
+
+  const MeshTopology * top = &mesh.GetTopology();
+  int npoints = mesh.GetNP();
+  int nelements = mesh.GetNE();
+  int nsurfelem = mesh.GetNSE();
+  int nedges = top->GetNEdges();
+  int i, j;
+  
+  int inverttets = mparam.inverttets;
+  int invertsurf = mparam.inverttrigs;
+  ARRAY<int> edges;
+
+  ofstream outfile (filename.c_str());
+
+  outfile.precision(6);
+  outfile.setf (ios::fixed, ios::floatfield);
+  outfile.setf (ios::showpoint);
+
+
+  // vertices with coordinates  
+  outfile << npoints << "\n";
+  for (i = 1; i <= npoints; i++)
+    {
+      const Point3d & p = mesh.Point(i);
+      
+      outfile.width(10);
+      outfile << p.X() << " ";
+      outfile.width(9);
+      outfile << p.Y() << " ";
+      outfile.width(9);
+      outfile << p.Z() << "\n";
+    }
+
+  // element - edge - list
+  outfile << nelements << " " << nedges << "\n";
+  for (i = 1; i <= nelements; i++)
+    {
+      Element el = mesh.VolumeElement(i);
+      if (inverttets)
+      	el.Invert();
+      outfile.width(4);
+      outfile << el.GetIndex() << "  ";
+      outfile.width(8);
+      outfile << el.GetNP();
+      for (j = 1; j <= el.GetNP(); j++)
+	{
+	  outfile << " ";
+	  outfile.width(8);
+	  outfile << el.PNum(j);
+	}
+
+      top->GetElementEdges(i,edges);
+      outfile << endl << "      ";
+      outfile.width(8);
+      outfile << edges.Size();
+      for (j=1; j <= edges.Size(); j++)
+	{
+	  outfile << " ";
+	  outfile.width(8);
+	  outfile << edges[j-1];
+	}
+      outfile << "\n";
+
+      // orientation:
+      top->GetElementEdgeOrientations(i,edges);
+      outfile << "              ";
+      for (j=1; j <= edges.Size(); j++)
+	{
+	  outfile << " ";
+	  outfile.width(8);
+	  outfile << edges[j-1];
+	}
+      outfile << "\n";
+    }
+
+  // surface element - edge - list (with boundary conditions)
+  outfile << nsurfelem << "\n";
+  for (i = 1; i <= nsurfelem; i++)
+    {
+      Element2d el = mesh.SurfaceElement(i);
+      if (invertsurf)
+	el.Invert();
+      outfile.width(4);
+      outfile << mesh.GetFaceDescriptor (el.GetIndex()).BCProperty() << "  ";
+      outfile.width(8);
+      outfile << el.GetNP();
+      for (j = 1; j <= el.GetNP(); j++)
+	{
+	  outfile << " ";
+	  outfile.width(8);
+	  outfile << el.PNum(j);
+	}
+
+      top->GetSurfaceElementEdges(i,edges);
+      outfile << endl << "      ";
+      outfile.width(8);
+      outfile << edges.Size();
+      for (j=1; j <= edges.Size(); j++)
+	{
+	  outfile << " ";
+	  outfile.width(8);
+	  outfile << edges[j-1];
+	}
+      outfile << "\n";
+    }
+
+
+  int v1, v2;
+  // edge - vertex - list
+  outfile << nedges << "\n";
+  for (i=1; i <= nedges; i++)
+    {
+      top->GetEdgeVertices(i,v1,v2);
+      outfile.width(4);
+      outfile << v1;
+      outfile << " ";
+      outfile.width(8);
+      outfile << v2 << endl;
+    }
+}
 
 
 
diff --git a/Netgen/libsrc/interface/writeuser.hpp b/Netgen/libsrc/interface/writeuser.hpp
index 26e8f8eeb1..afb2a6d5e3 100644
--- a/Netgen/libsrc/interface/writeuser.hpp
+++ b/Netgen/libsrc/interface/writeuser.hpp
@@ -20,7 +20,7 @@ void WriteFile (int typ,
 
 extern 
 void ReadFile (Mesh & mesh,
-	       const char * filename);
+	       const string & filename);
 
 extern 
 void ImportSolution (const char * filename);
@@ -54,14 +54,16 @@ void WriteFEPPFormat (const Mesh & mesh,
 		      const CSGeometry & geom,
 		      const string & filename);
 
+extern
+void WriteGmshFormat (const Mesh & mesh,
+                         const CSGeometry & geom,
+                         const string & filename);
+
 extern
 void WriteUserChemnitz (const Mesh & mesh,
 			const string & filename);
 
 
-
-
-
 extern 
 void WriteDiffPackFormat (const Mesh & mesh,
 			  const CSGeometry & geom,
@@ -92,6 +94,17 @@ extern
 void WriteFEAPFormat (const Mesh & mesh,
 		      const string & filename);
 
+extern
+void WriteElmerFormat (const Mesh & mesh,
+		       const string & filename);
+
+
+extern
+void WriteEdgeElementFormat (const Mesh & mesh,
+			     const CSGeometry & geom,
+			     const string & filename);
+
+
 
 extern void RegisterUserFormats (ARRAY<const char*> & names);
 extern bool WriteUserFormat (const string & format, 	
diff --git a/Netgen/libsrc/interface/wuchemnitz.cpp b/Netgen/libsrc/interface/wuchemnitz.cpp
index 306cbe6268..82a513d1ca 100644
--- a/Netgen/libsrc/interface/wuchemnitz.cpp
+++ b/Netgen/libsrc/interface/wuchemnitz.cpp
@@ -225,7 +225,7 @@ void Convert ()
           i2.I2() = faces.Get(i).p2;
           break;
         }
-      if (i2.I1() > i2.I2()) Swap (i2.I1(), i2.I2());
+      if (i2.I1() > i2.I2()) swap (i2.I1(), i2.I2());
       if (edgeindex.Used (i2)) 
         edgei = edgeindex.Get(i2);
       else
diff --git a/Netgen/libsrc/linalg/Makefile b/Netgen/libsrc/linalg/Makefile
index a3ca3763d9..0cb19b0932 100644
--- a/Netgen/libsrc/linalg/Makefile
+++ b/Netgen/libsrc/linalg/Makefile
@@ -1,7 +1,7 @@
 #
 # Makefile for linear algebra library
 #
-src = densemat.cpp vector.cpp polynomial.cpp
+src = basemat.cpp densemat.cpp vector.cpp sparsmat.cpp polynomial.cpp
 #
 lib = la
 libpath = libsrc/linalg
diff --git a/Netgen/libsrc/linalg/basemat.cpp b/Netgen/libsrc/linalg/basemat.cpp
index 49fd4c2e41..889d79d668 100644
--- a/Netgen/libsrc/linalg/basemat.cpp
+++ b/Netgen/libsrc/linalg/basemat.cpp
@@ -1,3 +1,5 @@
+#ifdef ABC
+
 #include <mystdlib.h>
 #include <linalg.hpp>
 
@@ -71,8 +73,8 @@ ostream & BaseMatrix :: Print (ostream & s) const
   }
 
 
-
-TempVector BaseMatrix :: operator* (const BaseVector & v) const
+  /*
+TempVector BaseMatrix :: operator* (const Vector & v) const
   {
   Vector * prod = new Vector(Height());
 
@@ -91,7 +93,7 @@ TempVector BaseMatrix :: operator* (const BaseVector & v) const
 
   return *prod;
   }
-
+  */
 
 
 DenseMatrix operator* (const BaseMatrix & m1, const BaseMatrix & m2)
@@ -147,14 +149,17 @@ DenseMatrix operator+ (const BaseMatrix & m1, const BaseMatrix & m2)
   }
 
 
-void BaseMatrix :: Mult (const BaseVector & /* v */,
-      BaseVector & /* prod */) const
+void BaseMatrix :: Mult (const FlatVector & /* v */,
+			 FlatVector & /* prod */) const
   {
-  (*myerr) << "BaseMatrix :: Mult called" << endl;
+    (*myerr) << "BaseMatrix :: Mult called" << endl;
+    double * x = 0;
+    *x = 1;
+    assert (1);
   }
 
-void BaseMatrix :: MultTrans (const BaseVector &  v,
-      BaseVector & prod) const
+void BaseMatrix :: MultTrans (const Vector &  v,
+      Vector & prod) const
   {
   if (Symmetric())
     Mult (v, prod);
@@ -162,16 +167,16 @@ void BaseMatrix :: MultTrans (const BaseVector &  v,
     (*myerr) << "BaseMatrix :: MultTrans called for non symmetric matrix" << endl;
   }
 
-void BaseMatrix :: Residuum (const BaseVector &  x,
-      const BaseVector & b, BaseVector & res) const
+void BaseMatrix :: Residuum (const Vector &  x,
+      const Vector & b, Vector & res) const
   {
   Mult (x, res);
   res *= -1;
   res.Add (1, b);
   }
 
-void BaseMatrix :: ResiduumTrans (const BaseVector & x,
-      const BaseVector & b, BaseVector & res) const
+void BaseMatrix :: ResiduumTrans (const Vector & x,
+      const Vector & b, Vector & res) const
   {
   MultTrans (x, res);
   res *= -1;
@@ -185,7 +190,7 @@ BaseMatrix * BaseMatrix :: Copy () const
   }
 
 
-BaseVector * BaseMatrix :: CreateVector () const
+Vector * BaseMatrix :: CreateVector () const
   {
   return new Vector (Height());
   }
@@ -296,14 +301,14 @@ void BaseMatrix :: SolveDestroy (const Vector & v, Vector & sol)
     (*myerr) << "SolveDestroy: Matrix not square";
     return;
     }
-  if (Width() != v.Length())
+  if (Width() != v.Size())
     {
     (*myerr) << "SolveDestroy: Matrix and Vector don't fit";
     return;
     }
 
   sol = v;
-  if (Height() != sol.Length())
+  if (Height() != sol.Size())
     {
     (*myerr) << "SolveDestroy: Solution Vector not ok";
     return;
@@ -324,12 +329,12 @@ void BaseMatrix :: SolveDestroy (const Vector & v, Vector & sol)
 
   for (i = Height(); i >= 1; i--)
     {
-    q = sol(i);
+    q = sol.Elem(i);
     for (j = i+1; j <= Height(); j++)
       {
       q -= (*this)(i,j) * sol.Get(j);
       }
-    sol.Set(i, q / (*this)(i,i));
+    sol.Elem(i) = q / (*this)(i,i);
     }
   }
 
@@ -349,28 +354,29 @@ void BaseMatrix :: Solve (const Vector & v, Vector & sol) const
   }
 
 
-Vector BaseMatrix :: SolveDestroyFunc (const Vector & /* b */) const
+  /*
+Vector BaseMatrix :: SolveDestroyFunc (const Vector & b) const
 {
   return Vector(0);
 }
-
+*/
 
 
 Vector BaseMatrix :: Solve (const Vector & v) const
   {
-  Vector sol (v.Length());
+  Vector sol (v.Size());
 
   if (Width() != Height())
     {
     (*myerr) << "Solve: Matrix not square";
     return v;
     }
-  if (Width() != v.Length())
+  if (Width() != v.Size())
     {
     (*myerr) << "Solve: Matrix and Vector don't fit";
     return v;
     }
-  if (Width() != sol.Length())
+  if (Width() != sol.Size())
     {
     (*myerr) << "Solve: Vector sol not allocated" << endl;
     }
@@ -463,3 +469,4 @@ BaseMatrix * BaseMatrix :: InverseMatrix (const BitArray * /* inner */) const
 
 
 }
+#endif
diff --git a/Netgen/libsrc/linalg/basemat.hpp b/Netgen/libsrc/linalg/basemat.hpp
index 89a9e21411..d47abe790d 100644
--- a/Netgen/libsrc/linalg/basemat.hpp
+++ b/Netgen/libsrc/linalg/basemat.hpp
@@ -1,3 +1,5 @@
+#ifdef NONE
+
 #ifndef FILE_BASEMAT
 #define FILE_BASEMAT
 
@@ -55,32 +57,32 @@ public:
   virtual ostream & Print (ostream & s) const;
 
   ///
-  TempVector operator* (const BaseVector & v) const;
+  // TempVector operator* (const Vector & v) const;
 
 
   ///
-  virtual void Mult (const BaseVector & v, BaseVector & prod) const;
+  virtual void Mult (const FlatVector & v, FlatVector & prod) const;
   ///
-  virtual void MultTrans (const BaseVector & v, BaseVector & prod) const;
+  virtual void MultTrans (const Vector & v, Vector & prod) const;
   ///
-  virtual void Residuum (const BaseVector & x, const BaseVector & b, BaseVector & res) const;
+  virtual void Residuum (const Vector & x, const Vector & b, Vector & res) const;
   ///
-  virtual void ResiduumTrans (const BaseVector & x, const BaseVector & b, BaseVector & res) const;
-  //  virtual double EvaluateBilinearform (const BaseVector & x);
+  virtual void ResiduumTrans (const Vector & x, const Vector & b, Vector & res) const;
+  //  virtual double EvaluateBilinearform (const Vector & x);
 
   virtual BaseMatrix * Copy () const;
   ///
-  virtual BaseVector * CreateVector () const;
+  virtual Vector * CreateVector () const;
 
   ///
   virtual void AddElementMatrix (const ARRAY<INDEX> & /* pnum */, 
 				 const BaseMatrix & /* elemmat */) { };
   ///
   virtual void MultElementMatrix (const ARRAY<INDEX> & /* pnum */, 
-				  const BaseVector & /* x */, BaseVector & /* y */) { };
+				  const Vector & /* x */, Vector & /* y */) { };
   ///
   virtual void MultTransElementMatrix (const ARRAY<INDEX> & /* pnum */, 
-				       const BaseVector & /* x */, BaseVector & /* y */) { };
+				       const Vector & /* x */, Vector & /* y */) { };
 
 
   ///
@@ -94,7 +96,7 @@ public:
   ///
   void Solve (const Vector & b, Vector & x) const;
   ///
-  virtual Vector SolveDestroyFunc (const Vector & b) const;
+  // virtual Vector SolveDestroyFunc (const Vector & b) const;
   ///
   Vector Solve (const Vector & b) const;
   ///
@@ -102,4 +104,6 @@ public:
 };
 
 
+#endif
+
 #endif
diff --git a/Netgen/libsrc/linalg/densemat.hpp b/Netgen/libsrc/linalg/densemat.hpp
index f2dc9bd605..fadb128d6a 100644
--- a/Netgen/libsrc/linalg/densemat.hpp
+++ b/Netgen/libsrc/linalg/densemat.hpp
@@ -39,6 +39,8 @@ public:
 
   double & operator() (int i, int j) { return data[i*width+j]; }
   double operator() (int i, int j) const { return data[i*width+j]; }
+  double & operator() (int i) { return data[i]; }
+  double operator() (int i) const { return data[i]; }
 
   ///
   DenseMatrix & operator= (const DenseMatrix & m2);
@@ -59,6 +61,7 @@ public:
     const double * mp, * sp;
     double * dp;
     
+#ifdef DEBUG
     if (prod.Size() != height)
       {
 	cerr << "Mult: wrong vector size " << endl;
@@ -66,7 +69,7 @@ public:
 	// prod.SetSize (height);
       }
     
-#ifdef DEBUG
+
     if (!height) 
       {
 	cout << "DenseMatrix::Mult height = 0" << endl;
@@ -183,21 +186,18 @@ public:
       }
   }
 
+  ///
   int Height() const { return height; }
-  int Width() const { return WIDTH; }
 
-  /*
-  ///
-  virtual double & operator() (INDEX i, INDEX j);
   ///
-  virtual double operator() (INDEX i, INDEX j) const;
-  */
+  int Width() const { return WIDTH; }
 
   ///
   MatrixFixWidth & operator= (double v)
   {
     for (int i = 0; i < height*WIDTH; i++)
-      data[i] = 0; 
+      data[i] = v; 
+    return *this;
   }
 
   ///
diff --git a/Netgen/libsrc/linalg/linalg.hpp b/Netgen/libsrc/linalg/linalg.hpp
index 3dd73e8ef0..4599fad48d 100644
--- a/Netgen/libsrc/linalg/linalg.hpp
+++ b/Netgen/libsrc/linalg/linalg.hpp
@@ -25,7 +25,9 @@
 namespace netgen
 {
 #include "vector.hpp"
+  // #include "basemat.hpp"
 #include "densemat.hpp"
+  // #include "sparsmat.hpp"
 #include "polynomial.hpp"
 }
 #endif
diff --git a/Netgen/libsrc/linalg/sparsmat.cpp b/Netgen/libsrc/linalg/sparsmat.cpp
index cb2f32a233..f3249f82e6 100644
--- a/Netgen/libsrc/linalg/sparsmat.cpp
+++ b/Netgen/libsrc/linalg/sparsmat.cpp
@@ -1,3 +1,6 @@
+//#include <algorithm>
+#ifdef ABC
+
 #include <mystdlib.h>
  
 #include <linalg.hpp>
@@ -165,7 +168,7 @@ SparseMatrix :: SparseMatrix (INDEX h, INDEX w)
 
 
 
-void SparseMatrix :: Mult (const BaseVector & bv, BaseVector & bprod) const
+void SparseMatrix :: Mult (const Vector & v, Vector & prod) const
   {
   double sum, vi;
   INDEX i, j, n;
@@ -173,13 +176,13 @@ void SparseMatrix :: Mult (const BaseVector & bv, BaseVector & bprod) const
   const INDEX * col;
   const double * valp;
 
-  const Vector & v = bv.CastToVector();
-  Vector & prod = bprod.CastToVector();
+  //  const Vector & v = bv.CastToVector();
+  //  Vector & prod = bprod.CastToVector();
 
-  prod.SetLength (Height());
+  prod.SetSize (Height());
 
   /*
-  if (prod.Length() != Height() || v.Length() != Width())
+  if (prod.Size() != Height() || v.Size() != Width())
     {
     (*myerr) << "SparseMatrix::Mult: Dimensions don't fit" << endl;
     return;
@@ -229,7 +232,7 @@ void SparseMatrix :: Mult (const BaseVector & bv, BaseVector & bprod) const
 
 	  
 	  linep++;
-	  prod.Set (i, sum);
+	  prod.Elem(i) = sum;
 	}
 
     }
@@ -261,18 +264,18 @@ void SparseMatrix :: Mult (const BaseVector & bv, BaseVector & bprod) const
     }
   }
 
-void SparseMatrix :: MultTrans (const BaseVector & bv, BaseVector & bprod) const
+void SparseMatrix :: MultTrans (const Vector & v, Vector & prod) const
   {
   INDEX i, j, n, coln;
-  const Vector & v = bv.CastToVector();
-  Vector & prod = bprod.CastToVector();
+  //  const Vector & v = bv.CastToVector();
+  //  Vector & prod = bprod.CastToVector();
   const INDEX * col;
   const double * valp;
   double val;
 
-  prod.SetLength (Width());
+  prod.SetSize (Width());
 
-  if (prod.Length() != Width() || v.Length() != Height())
+  if (prod.Size() != Width() || v.Size() != Height())
     {
     (*myerr) << "SparseMatrix::Mult: Dimensions don't fit" << endl;
     return;
@@ -307,8 +310,8 @@ void SparseMatrix :: MultTrans (const BaseVector & bv, BaseVector & bprod) const
 
 
 
-void SparseMatrix :: Residuum (const BaseVector & bx, const BaseVector & bb,
-      BaseVector & bres) const
+void SparseMatrix :: Residuum (const Vector & bx, const Vector & bb,
+      Vector & bres) const
   {
     BaseMatrix :: Residuum (bx, bb, bres);
     /*
@@ -320,10 +323,10 @@ void SparseMatrix :: Residuum (const BaseVector & bx, const BaseVector & bb,
   const Vector & b = bb.CastToVector();
   Vector & res = bres.CastToVector();
 
-  res.SetLength (b.Length());
+  res.SetSize (b.Size());
 
-  if (res.Length() != b.Length() || b.Length() != Height() ||
-      x.Length() != Width())
+  if (res.Size() != b.Size() || b.Size() != Height() ||
+      x.Size() != Width())
     {
     (*myerr) << "SparseMatrix::Residuum: Dimensions don't fit" << endl;
     return;
@@ -368,9 +371,9 @@ void SparseMatrix :: Residuum (const BaseVector & bx, const BaseVector & bb,
   }
 
 
-void SparseMatrix :: ResiduumTrans (const BaseVector & /* bx */, 
-				    const BaseVector & /* bb */,
-				    BaseVector & /* bres */) const
+void SparseMatrix :: ResiduumTrans (const Vector & /* bx */, 
+				    const Vector & /* bb */,
+				    Vector & /* bres */) const
   {
     cerr << "SparseMastrix :: Residuumtrans called, but not implemented" << endl;
 
@@ -383,10 +386,10 @@ void SparseMatrix :: ResiduumTrans (const BaseVector & /* bx */,
   Vector & res = bres.CastToVector();
 
 
-  res.SetLength (Width());
+  res.SetSize (Width());
 
-  if (res.Length() != b.Length() || b.Length() != Width() ||
-      x.Length() != Height())
+  if (res.Size() != b.Size() || b.Size() != Width() ||
+      x.Size() != Height())
     {
     (*myerr) << "SparseMatrix::ResiduumTrans: Dimensions don't fit" << endl;
     return;
@@ -534,7 +537,7 @@ double SparseMatrix :: RowTimesVector (INDEX i, const Vector & v) const
   int j;
 
   /*
-  if (Width() > v.Length())
+  if (Width() > v.Size())
     {
     cerr << "SparseMatrix::RowTimesVector: Size doesn't fit" << endl;
     return 0;
@@ -578,10 +581,10 @@ void SparseMatrix :: AddRowToVector (INDEX i, double s, Vector & v) const
   int coln, j;
   
 #ifdef debug  
-  if (Width() > v.Length())
+  if (Width() > v.Size())
     {
       cerr << "SparseMatrix::AddRowToVector: Size doesn't fit" 
-	   << "w = " << Width() << " len = " << v.Length() << endl;
+	   << "w = " << Width() << " len = " << v.Size() << endl;
       return;
     }
 #endif    
@@ -1550,7 +1553,7 @@ double SparseMatrix :: RowTimesVector (INDEX i, const Vector & v) const
   double sum;
   int j;
 
-  if (Width() > v.Length())
+  if (Width() > v.Size())
     {
     cerr << "SparseMatrix::RowTimesVector: Size doesn't fit" << endl;
     return 0;
@@ -1572,10 +1575,10 @@ void SparseMatrix :: AddRowToVector (INDEX i, double s, Vector & v) const
   const colstruct * col;
   int j;
 
-  if (Width() > v.Length())
+  if (Width() > v.Size())
     {
     cerr << "SparseMatrix::AddRowToVector: Size doesn't fit" 
-    	 << "w = " << Width() << " len = " << v.Length() << endl;
+    	 << "w = " << Width() << " len = " << v.Size() << endl;
     return;
     }
 
@@ -1701,3 +1704,4 @@ void SparseMatrix :: SetGraph (const class TABLE<INDEX> & graph)
 
 #endif
 }
+#endif
diff --git a/Netgen/libsrc/linalg/sparsmat.hpp b/Netgen/libsrc/linalg/sparsmat.hpp
index 884ab6575c..74aaa3134e 100644
--- a/Netgen/libsrc/linalg/sparsmat.hpp
+++ b/Netgen/libsrc/linalg/sparsmat.hpp
@@ -1,3 +1,5 @@
+#ifdef NONE
+
 #ifndef FILE_SPARSMAT
 #define FILE_SPARSMAT
 
@@ -86,15 +88,15 @@ public:
 
 
   /// prod = matrix x v
-  virtual void Mult (const BaseVector & v, BaseVector & prod) const;
+  virtual void Mult (const Vector & v, Vector & prod) const;
   /// prod = matrix^T x v
-  virtual void MultTrans (const BaseVector & v, BaseVector & prod) const;
+  virtual void MultTrans (const Vector & v, Vector & prod) const;
   /// res = b - mat x x
-  virtual void Residuum (const BaseVector & x, const BaseVector & b, 
-			 BaseVector & res) const;
+  virtual void Residuum (const Vector & x, const Vector & b, 
+			 Vector & res) const;
   /// res = b - mat^T x x
-  virtual void ResiduumTrans (const BaseVector & x, const BaseVector & b,
-			      BaseVector & res) const;
+  virtual void ResiduumTrans (const Vector & x, const Vector & b,
+			      Vector & res) const;
 
 
   /**
@@ -258,4 +260,6 @@ protected:
 };
 
 
+#endif
+
 #endif
diff --git a/Netgen/libsrc/linalg/vector.cpp b/Netgen/libsrc/linalg/vector.cpp
index 47be4eefd9..05b5a0a71e 100644
--- a/Netgen/libsrc/linalg/vector.cpp
+++ b/Netgen/libsrc/linalg/vector.cpp
@@ -1,6 +1,7 @@
 #ifdef abc
 #include <mystdlib.h>
 #include <linalg.hpp>
+#include <algorithm>
 
 namespace netgen
 {
diff --git a/Netgen/libsrc/makefile.inc b/Netgen/libsrc/makefile.inc
index 6a98d84e43..76369d5fb6 100644
--- a/Netgen/libsrc/makefile.inc
+++ b/Netgen/libsrc/makefile.inc
@@ -7,10 +7,10 @@
 CPP_DIR=../..
 LIBSRC_DIR=$(CPP_DIR)/libsrc
 LIB_DIR=$(CPP_DIR)/lib/$(MACHINE)
+
 OCC_DIR=../../occ
-# OCC_DIR=/opt/OpenCASCADE5.1/ros
 OCCINC_DIR=$(OCC_DIR)/inc
-# OCCLIB_DIR=$(OCC_DIR)/lib
+OCCLIB_DIR=$(OCC_DIR)/lib
 #
 include $(LIBSRC_DIR)/makefile.mach.$(MACHINE)
 #
@@ -20,11 +20,13 @@ ARFLAGS = r
 #
 LIBB=$(LIB_DIR)/lib$(lib).a
 #
-.PRECIOUS: .cpp .hh 
-.SUFFIXES: .cpp .o 
+.PRECIOUS: .cpp .c
+.SUFFIXES: .cpp .c .o 
 #
 .cpp.o:
 	$(CPLUSPLUS) $(CPLUSPLUSFLAGS1) $(CPLUSPLUSFLAGS2) $(CPLUSPLUSFLAGSLIBRARY) $<
+.c.o:
+	$(CPLUSPLUS) $(CPLUSPLUSFLAGS1) $(CPLUSPLUSFLAGS2) $(CPLUSPLUSFLAGSLIBRARY) $<
 #
 #
 $(LIBB):: $(LIB_DIR) 
diff --git a/Netgen/libsrc/makefile.mach.FREEBSD b/Netgen/libsrc/makefile.mach.FREEBSD
new file mode 100644
index 0000000000..6f903d4e60
--- /dev/null
+++ b/Netgen/libsrc/makefile.mach.FREEBSD
@@ -0,0 +1,26 @@
+#
+# Machine dependent make include file for gcc
+# 
+#
+#CC=gcc
+CPLUSPLUS=$(CC)
+AR=ar
+LINK=$(CC)
+#MAKE=make
+RM=rm
+RANLIB=ranlib
+#
+# Machine dependent flags:
+#
+include $(LOCALBASE)/lib/tixConfig.sh
+include $(LOCALBASE)/lib/tcl$(TCL_VER)/tclConfig.sh
+include $(LOCALBASE)/lib/tk$(TK_VER)/tkConfig.sh
+tcltklib = `echo $(TIX_BUILD_LIB_SPEC)` `echo $(TK_LIB_SPEC)` `echo $(TCL_LIB_FLAG)`
+
+CFLAGS2 =
+CPLUSPLUSFLAGS2 = $(CXXFLAGS) -I$(X11BASE)/include -DLINUX -DOPENGL
+CPLUSPLUSFLAGS3 = -I$(LIBSRC_DIR)/step `echo $(TCL_INCLUDE_SPEC)` `echo -I$(TK_PREFIX)`/include/tk`echo $(TK_VERSION)`
+#
+LINKFLAGS2 =  -L$(LOCALBASE)/lib -L$(X11BASE)/lib
+
+SYSLIB2 = -lstdc++
diff --git a/Netgen/libsrc/makefile.mach.INTEL b/Netgen/libsrc/makefile.mach.INTEL
index c445191a24..deef5b091c 100644
--- a/Netgen/libsrc/makefile.mach.INTEL
+++ b/Netgen/libsrc/makefile.mach.INTEL
@@ -1,5 +1,5 @@
 #
-# Machine dependent make include file for Intel compiler
+# Machine dependent make include file
 #
 CC=icc
 CPLUSPLUS=$(CC)
@@ -12,7 +12,23 @@ RANLIB=ranlib
 # Machine dependent flags:
 #
 CFLAGS2 =
-CPLUSPLUSFLAGS2 = -O2  -wd654 -DLINUX -DOPENGL 
+CPLUSPLUSFLAGS2 = -O2 -wd1572 -Ob2 -DLINUX -DOPENGL -DNGSOLVE \
+	-Qoption,c,-ip_ninl_max_stats=1000 \
+	-Qoption,c,-ip_ninl_min_stats=50 \
+	-Qoption,c,-ip_ninl_max_total_stats=1000 \
+	-gcc-name=/usr/bin/g++
 #  
-LINKFLAGS2 =  -L/usr/openwin/lib -L/usr/X11R6/lib -L/usr/lib/GL3.5 
+LINKFLAGS2 =  -L/usr/openwin/lib -L/usr/X11R6/lib -L/usr/lib/GL3.5 \
+	-gcc-name=/usr/bin/g++
+#
+# SYSLIB2 = /opt/experimental/lib/libstdc++.a
+# SYSLIB2 = -lstdc++ 
+# -lgcc_s
+# SYSLIB2 = -L/usr/lib/lapack -lblas -lstdc++
+
+# goalngs = goalngs
+
+
+goalngs=goalngs
 
+appngs =  lib/$(MACHINE)/*.o -lngsolvebasic
\ No newline at end of file
diff --git a/Netgen/libsrc/makefile.mach.LINUX b/Netgen/libsrc/makefile.mach.LINUX
index 44e4a5ccdd..f5df356167 100644
--- a/Netgen/libsrc/makefile.mach.LINUX
+++ b/Netgen/libsrc/makefile.mach.LINUX
@@ -1,7 +1,9 @@
 #
-# Machine dependent make include file for gcc
+# Machine dependent make include file
 # 
 #
+# CC=/opt/gcc-dev/bin/gcc 
+# CC=/usr/local/bin/gcc
 CC=gcc
 CPLUSPLUS=$(CC)
 AR=ar
@@ -13,8 +15,17 @@ RANLIB=ranlib
 # Machine dependent flags:
 #
 CFLAGS2 =
-CPLUSPLUSFLAGS2 = -O2 -I/usr/X11R6/include -DLINUX -DOPENGL 
-#
-LINKFLAGS2 =  -L/usr/openwin/lib -L/usr/X11R6/lib
 
-SYSLIB2 = -lstdc++
+CPLUSPLUSFLAGS2 = -O2 -I/usr/include/GL3.5 -DLINUX -DOPENGL \
+	-ftemplate-depth-99 -finline-limit=20000 \
+	-funroll-loops  -DNGSOLVE
+
+LINKFLAGS2 =   -L/usr/openwin/lib -L/usr/X11R6/lib -L/usr/lib/GL3.5 -lstdc++ 
+
+
+goalngs=goalngs
+
+# lapack =  -llapack  -lblas -lgmp -lg2c
+
+
+appngs =  lib/$(MACHINE)/*.o -lngsolvebasic
\ No newline at end of file
diff --git a/Netgen/libsrc/makefile.mach.LINUXGCC33 b/Netgen/libsrc/makefile.mach.LINUXGCC33
new file mode 100644
index 0000000000..0a6ee84afb
--- /dev/null
+++ b/Netgen/libsrc/makefile.mach.LINUXGCC33
@@ -0,0 +1,33 @@
+#
+# Machine dependent make include file
+# 
+#
+# CC=/opt/gcc33/bin/gcc 
+# CC=/usr/local/bin/gcc
+CC=gcc
+CPLUSPLUS=$(CC)
+AR=ar
+LINK=$(CC)
+MAKE=make
+RM=rm
+RANLIB=ranlib
+#
+#
+CFLAGS2 =
+CPLUSPLUSFLAGS2 = -O2 -DLINUX -DOPENGL \
+	-ftemplate-depth-99 -finline-limit=20000 \
+	-mcpu=pentium4  -fforce-addr  -funroll-loops \
+	-DnoTRAFO -DNGSOLVE -DnoADDON -DnoPML -DnoLAPACK \
+	-DnoOCCGEOMETRY -I/usr/include/g++/backward -I./occ/inc -DnoDEBUG 
+#	
+#
+#  
+LINKFLAGS2 = -L/usr/openwin/lib -L/usr/X11R6/lib -L/usr/lib/GL3.5 
+
+SYSLIB2 =  -lstdc++ 
+
+goalngs   = goalngs
+
+appngs =  lib/$(MACHINE)/*.o -lngsolvebasic
+
+# occlib = -L$(OCCLIB_DIR) -lTKIGES -lTKBRep -lTKSTEP -lTKSTL
\ No newline at end of file
diff --git a/Netgen/libsrc/meshing/adfront2.cpp b/Netgen/libsrc/meshing/adfront2.cpp
index e273144501..01b591f477 100644
--- a/Netgen/libsrc/meshing/adfront2.cpp
+++ b/Netgen/libsrc/meshing/adfront2.cpp
@@ -142,6 +142,8 @@ INDEX AdFront2 :: AddLine (INDEX pi1, INDEX pi2,
   FrontPoint2 & p1 = points.Elem(pi1);
   FrontPoint2 & p2 = points.Elem(pi2);
 
+
+
   nfl++;
 
   p1.AddLine();
@@ -265,7 +267,8 @@ void AdFront2 :: ResetClass (INDEX li)
 
 int AdFront2 :: SelectBaseLine (Point3d & p1, Point3d & p2, 
 				const PointGeomInfo *& geominfo1,
-				const PointGeomInfo *& geominfo2)
+				const PointGeomInfo *& geominfo2,
+				int & qualclass)
 {
   int i, hi;
 
@@ -315,7 +318,7 @@ int AdFront2 :: SelectBaseLine (Point3d & p1, Point3d & p2,
   
   if (!baselineindex)
     {
-      (*testout) << "nfl = " << nfl << " tot l = " << lines.Size() << endl;
+      // (*testotu) << "nfl = " << nfl << " tot l = " << lines.Size() << endl;
       minval = INT_MAX;
       for (i = 1; i <= lines.Size(); i++)
 	if (lines.Get(i).Valid())
@@ -339,6 +342,9 @@ int AdFront2 :: SelectBaseLine (Point3d & p1, Point3d & p2,
   p2 = points.Get(lines.Get(baselineindex).L().I2()).P();
   geominfo1 = &lines.Get(baselineindex).GetGeomInfo(1);
   geominfo2 = &lines.Get(baselineindex).GetGeomInfo(2);
+
+  qualclass = lines.Get(baselineindex).LineClass();
+
   return baselineindex;
 }
 
@@ -367,24 +373,22 @@ int AdFront2 :: GetLocals (int baselineindex,
   static ARRAY<int> nearlines;
 
   nearlines.SetSize(0);
-  double dist = xh;
-  linesearchtree.GetIntersecting (p0 - Vec3d(dist, dist, dist),
-				  p0 + Vec3d(dist, dist, dist),
+
+  linesearchtree.GetIntersecting (p0 - Vec3d(xh, xh, xh),
+				  p0 + Vec3d(xh, xh, xh),
 				  nearlines);
 
-  //  for (i = 1; i <= lines.Size(); i++)
   for (ii = 1; ii <= nearlines.Size(); ii++)
     {
       i = nearlines.Get(ii);
 
       if (lines.Get(i).Valid() && i != baselineindex)
 	{
-	  const Point3d & p1 = points.Get(lines.Get(i).L().I1()).P();
-	  const Point3d & p2 = points.Get(lines.Get(i).L().I2()).P();
+	  // const Point3d & p1 = points.Get(lines.Get(i).L().I1()).P();
+	  // const Point3d & p2 = points.Get(lines.Get(i).L().I2()).P();
 
-	  midp = Center (p1, p2);
-	  
-	  if (Dist (midp, p0) <= xh + 0.5 * Dist (p1, p2))
+	  //	  midp = Center (p1, p2);
+	  //	  if (Dist (midp, p0) <= xh + 0.5 * Dist (p1, p2))
 	    {
 	      loclines.Append(lines.Get(i).L());
 	      lindex.Append(i);
@@ -396,8 +400,10 @@ int AdFront2 :: GetLocals (int baselineindex,
 
   invpindex.SetSize (points.Size());
   for (i = 1; i <= loclines.Size(); i++)
-    for (j = 1; j <= 2; j++)
-      invpindex.Elem(loclines.Get(i).I(j)) = 0;
+    {
+      invpindex.Elem(loclines.Get(i).I1()) = 0;
+      invpindex.Elem(loclines.Get(i).I2()) = 0;
+    }
 
   for (i = 1; i <= loclines.Size(); i++)
     {
@@ -454,22 +460,10 @@ int AdFront2 :: GetLocals (int baselineindex,
   for (i = 1; i <= locpoints.Size(); i++)
     {
       int pi = pindex.Get(i);
-
-
       
       if (points.Get(pi).mgi)
 	for (j = 1; j <= points.Get(pi).mgi->GetNPGI(); j++)
 	  pgeominfo.Elem(i).AddPointGeomInfo (points.Get(pi).mgi->GetPGI(j));
-/*
-	{
-	  for (j = 0; j < points.Get(pi).mgi->cnt; j++)
-	    {
-	      pgeominfo.Elem(i).mgi[pgeominfo.Elem(i).cnt] = 
-		points.Get(pi).mgi->mgi[j];
-	      pgeominfo.Elem(i).cnt++;
-	    }
-	}
-*/
     }
 	
   
@@ -485,7 +479,6 @@ int AdFront2 :: GetLocals (int baselineindex,
       }
   */
 
-
   if (loclines.Size() == 1)
     {
       cout << "loclines.Size = 1" << endl;
@@ -495,8 +488,6 @@ int AdFront2 :: GetLocals (int baselineindex,
 		 << " p0 = " << p0 << endl;
     }
 
-
-
   return lines.Get(baselineindex).LineClass();
 }
 
diff --git a/Netgen/libsrc/meshing/adfront2.hpp b/Netgen/libsrc/meshing/adfront2.hpp
index b7a45e34e8..4e72c087fb 100644
--- a/Netgen/libsrc/meshing/adfront2.hpp
+++ b/Netgen/libsrc/meshing/adfront2.hpp
@@ -192,7 +192,8 @@ public:
   ///
   int SelectBaseLine (Point3d & p1, Point3d & p2, 
 		      const PointGeomInfo *& geominfo1,
-		      const PointGeomInfo *& geominfo2);
+		      const PointGeomInfo *& geominfo2,
+		      int & qualclass);
 
   ///
   int GetLocals (int baseline, 
diff --git a/Netgen/libsrc/meshing/adfront3.cpp b/Netgen/libsrc/meshing/adfront3.cpp
index 657d0bc12b..1bf686e103 100644
--- a/Netgen/libsrc/meshing/adfront3.cpp
+++ b/Netgen/libsrc/meshing/adfront3.cpp
@@ -9,7 +9,7 @@ namespace netgen
 
 FrontPoint3 :: FrontPoint3 () 
 { 
-  globalindex = 0;
+  globalindex = -1;
   nfacetopoint = 0; 
   frontnr = 1000; 
   cluster = 0;
@@ -47,7 +47,7 @@ FrontFace :: FrontFace (const Element2d & af)
 
 void FrontFace :: Invalidate ()
 { 
-  f.Delete();  // PNum(1) = 0; 
+  f.Delete(); 
   oldfront = 0; 
   qualclass = 1000; 
 }
@@ -67,9 +67,7 @@ AdFront3 :: AdFront3 ()
   hashon = 1;
   hashcreated = 0;
   if (hashon) 
-    {
-      hashtable.Init(&points, &faces);
-    }
+    hashtable.Init(&points, &faces);
 
   facetree = NULL;
   connectedpairs = NULL;
@@ -82,10 +80,8 @@ AdFront3 :: AdFront3 ()
 
 AdFront3 :: ~AdFront3 ()
 {
-  if (facetree)
-    delete facetree;
-  if (connectedpairs)
-    delete connectedpairs;
+  delete facetree;
+  delete connectedpairs;
 }
 
 void AdFront3 :: GetPoints (ARRAY<Point3d> & apoints) const
@@ -97,7 +93,7 @@ void AdFront3 :: GetPoints (ARRAY<Point3d> & apoints) const
 }
 
 
-INDEX AdFront3 :: AddPoint (const Point3d & p, PointIndex globind)
+PointIndex AdFront3 :: AddPoint (const Point3d & p, PointIndex globind)
 {
   if (delpointl.Size())
     {
@@ -177,23 +173,20 @@ INDEX AdFront3 :: AddFace (const Element2d & aface)
 
 void AdFront3 :: DeleteFace (INDEX fi)
 {
-  int i;
-  INDEX pi;
-
   nff--;
 
-  for (i = 1; i <= faces.Get(fi).Face().GetNP(); i++)
+  for (int i = 1; i <= faces.Get(fi).Face().GetNP(); i++)
     {
-      pi = faces.Get(fi).Face().PNum(i);
-      points.Elem(pi).RemoveFace();
-      if (!points.Elem(pi).Valid())
+      PointIndex pi = faces.Get(fi).Face().PNum(i);
+      points[pi].RemoveFace();
+      if (!points[pi].Valid())
 	delpointl.Append (pi);
     }
 
   const Element2d & face = faces.Get(fi).Face();
-  const Point3d & p1 = points.Get(face.PNum(1)).P();
-  const Point3d & p2 = points.Get(face.PNum(2)).P();
-  const Point3d & p3 = points.Get(face.PNum(3)).P();
+  const Point3d & p1 = points[face.PNum(1)].P();
+  const Point3d & p2 = points[face.PNum(2)].P();
+  const Point3d & p3 = points[face.PNum(3)].P();
 
   vol -= 1.0/6.0 * (p1.X() + p2.X() + p3.X()) *
     ( (p2.Y() - p1.Y()) * (p3.Z() - p1.Z()) -
@@ -201,7 +194,7 @@ void AdFront3 :: DeleteFace (INDEX fi)
 
   if (face.GetNP() == 4)
     {
-      const Point3d & p4 = points.Get(face.PNum(4)).P();      
+      const Point3d & p4 = points[face.PNum(4)].P();      
       vol -= 1.0/6.0 * (p1.X() + p3.X() + p4.X()) *
 	( (p3.Y() - p1.Y()) * (p4.Z() - p1.Z()) -
 	  (p3.Z() - p1.Z()) * (p4.Y() - p1.Y()) );
@@ -209,7 +202,6 @@ void AdFront3 :: DeleteFace (INDEX fi)
       nff4--;
     }
 
-
   faces.Elem(fi).Invalidate();
 }
 
@@ -217,11 +209,10 @@ void AdFront3 :: DeleteFace (INDEX fi)
 INDEX AdFront3 :: AddConnectedPair (const INDEX_2 & apair)
 {
   if (!connectedpairs)
-    connectedpairs = new TABLE<int> (GetNP());
+    connectedpairs = new TABLE<int, PointIndex::BASE> (GetNP());
 
-  //  (*testout) << "addconnectedpair " << apair << endl;
-  connectedpairs->Add1 (apair.I1(), apair.I2());
-  connectedpairs->Add1 (apair.I2(), apair.I1());
+  connectedpairs->Add (apair.I1(), apair.I2());
+  connectedpairs->Add (apair.I2(), apair.I1());
 
   return 0;
 }
@@ -265,9 +256,8 @@ void AdFront3 :: CreateTrees ()
 
   pmax = pmax + 0.5 * (pmax - pmin);
   pmin = pmin + 0.5 * (pmin - pmax);
-  if (facetree)
-    delete facetree;
 
+  delete facetree;
   facetree = new Box3dTree (pmin, pmax);
   
   for (i = 1; i <= GetNF(); i++)
@@ -312,16 +302,17 @@ void AdFront3 :: RebuildInternalTables ()
     if (faces.Get(i).Valid())
       {
 	hi++;
-	faces.Elem(hi) = faces.Get(i);
+	if (hi < i)
+	  faces.Elem(hi) = faces.Get(i);
       }
   
   faces.SetSize (nff);
 
   int np = points.Size();
 
-
-  for (i = 1; i <= np; i++)
-    points.Elem(i).cluster = i;
+  for (i = PointIndex::BASE; 
+       i < np+PointIndex::BASE; i++)
+    points[i].cluster = i;
 
   int change;
   do
@@ -331,12 +322,12 @@ void AdFront3 :: RebuildInternalTables ()
 	{
 	  const Element2d & el = faces.Get(i).Face();
 
-	  int mini = points.Get(el.PNum(1)).cluster;
+	  int mini = points[el.PNum(1)].cluster;
 	  int maxi = mini;
 	  
 	  for (j = 2; j <= 3; j++)
 	    {
-	      int ci = points.Get(el.PNum(j)).cluster;
+	      int ci = points[el.PNum(j)].cluster;
 	      if (ci < mini) mini = ci;
 	      if (ci > maxi) maxi = ci;
 	    }
@@ -345,37 +336,36 @@ void AdFront3 :: RebuildInternalTables ()
 	    {
 	      change = 1;
 	      for (j = 1; j <= 3; j++)
-		points.Elem(el.PNum(j)).cluster = mini;
+		points[el.PNum(j)].cluster = mini;
 	    }
 	}
     }
   while (change);
 
-  BitArray usecl(np);
+  BitArrayChar<PointIndex::BASE> usecl(np);
   usecl.Clear();
   for (i = 1; i <= faces.Size(); i++)
     {
-      usecl.Set (points.Get(faces.Get(i).Face().PNum(1)).cluster);
+      usecl.Set (points[faces.Get(i).Face().PNum(1)].cluster);
       faces.Elem(i).cluster =
-	points.Get(faces.Get(i).Face().PNum(1)).cluster;
+	points[faces.Get(i).Face().PNum(1)].cluster;
     }
   int cntcl = 0;
-  for (i = 1; i <= np; i++)
+  for (i = PointIndex::BASE; 
+       i < np+PointIndex::BASE; i++)
     if (usecl.Test(i))
       cntcl++;
 
-  ARRAY<double> clvol (np);
-  for (i = 1; i <= np; i++)
-    clvol.Elem(i) = 0;
+  ARRAY<double, PointIndex::BASE> clvol (np);
+  clvol = 0.0;
 
   for (i = 1; i <= faces.Size(); i++)
     {
       const Element2d & face = faces.Get(i).Face();
 
-      const Point3d & p1 = points.Get(face.PNum(1)).P();      
-      const Point3d & p2 = points.Get(face.PNum(2)).P();      
-      const Point3d & p3 = points.Get(face.PNum(3)).P();      
-
+      const Point3d & p1 = points[face.PNum(1)].P();      
+      const Point3d & p2 = points[face.PNum(2)].P();      
+      const Point3d & p3 = points[face.PNum(3)].P();      
       
       double vi = 1.0/6.0 * (p1.X() + p2.X() + p3.X()) *
 	( (p2.Y() - p1.Y()) * (p3.Z() - p1.Z()) -
@@ -383,31 +373,31 @@ void AdFront3 :: RebuildInternalTables ()
       
       if (face.GetNP() == 4)
 	{
-	  const Point3d & p4 = points.Get(face.PNum(4)).P();      
+	  const Point3d & p4 = points[face.PNum(4)].P();      
 	  vi += 1.0/6.0 * (p1.X() + p3.X() + p4.X()) *
 	    ( (p3.Y() - p1.Y()) * (p4.Z() - p1.Z()) -
 	      (p3.Z() - p1.Z()) * (p4.Y() - p1.Y()) );
 	}
      
-      clvol.Elem (faces.Get(i).cluster) += vi;
+      clvol[faces.Get(i).cluster] += vi;
     }
 
 
   int negvol = 0;
-  for (i = 1; i <= clvol.Size(); i++)
+  for (i = PointIndex::BASE; 
+       i < clvol.Size()+PointIndex::BASE; i++)
     {
-      if (clvol.Elem(i) < 0)
-	{
-	  negvol = 1;
-	}
+      if (clvol[i] < 0)
+	negvol = 1;
     }
 
   if (negvol)
     {
       for (i = 1; i <= faces.Size(); i++)
 	faces.Elem(i).cluster = 1;
-      for (i = 1; i <= points.Size(); i++)
-	points.Elem(i).cluster = 1;
+      for (i = PointIndex::BASE; 
+	   i < points.Size()+PointIndex::BASE; i++)
+	points[i].cluster = 1;
     }
 
   if (hashon) 
@@ -450,9 +440,9 @@ int AdFront3 :: SelectBaseElement ()
     if (faces.Elem(i).Valid())
       {
 	hi = faces.Get(i).QualClass() +
-	  points.Get(faces.Get(i).Face().PNum(1)).FrontNr() +
-	  points.Get(faces.Get(i).Face().PNum(2)).FrontNr() +
-	  points.Get(faces.Get(i).Face().PNum(3)).FrontNr();
+	  points[faces.Get(i).Face().PNum(1)].FrontNr() +
+	  points[faces.Get(i).Face().PNum(2)].FrontNr() +
+	  points[faces.Get(i).Face().PNum(3)].FrontNr();
 	
 	if (hi <= minval)
 	  {
@@ -469,9 +459,9 @@ int AdFront3 :: SelectBaseElement ()
 	if (faces.Elem(i).Valid())
 	  {
 	    hi = faces.Get(i).QualClass() +
-	      points.Get(faces.Get(i).Face().PNum(1)).FrontNr() +
-	      points.Get(faces.Get(i).Face().PNum(2)).FrontNr() +
-	      points.Get(faces.Get(i).Face().PNum(3)).FrontNr();
+	      points[faces.Get(i).Face().PNum(1)].FrontNr() +
+	      points[faces.Get(i).Face().PNum(2)].FrontNr() +
+	      points[faces.Get(i).Face().PNum(3)].FrontNr();
 	    
 	    if (hi <= minval)
 	      {
@@ -501,14 +491,16 @@ int AdFront3 :: GetLocals (int fstind,
   if (hashon && faces.Size() < 500) { hashon=0; }
   if (hashon && !hashcreated) 
     {
-      hashtable.Create(); hashcreated=1;
+      hashtable.Create(); 
+      hashcreated=1;
     }
 
   INDEX i, j;
-  INDEX pstind;
+  PointIndex pstind;
   INDEX pi;
   Point3d midp, p0;
-  static ARRAY<int> invpindex;
+
+  static ARRAY<int, PointIndex::BASE> invpindex;
   
   static ARRAY<Element2d> locfaces2;           //all local faces in radius xh
   static ARRAY<int> locfaces3;           // all faces in outer radius relh
@@ -521,7 +513,7 @@ int AdFront3 :: GetLocals (int fstind,
   int cluster = faces.Get(fstind).cluster;
 
   pstind = faces.Get(fstind).Face().PNum(1);
-  p0 = points.Get(pstind).P();
+  p0 = points[pstind].P();
   
   locfaces2.Append(faces.Get(fstind).Face());
   findex2.Append(fstind);
@@ -540,9 +532,9 @@ int AdFront3 :: GetLocals (int fstind,
 	  const Element2d & face = faces.Get(i).Face();
 	  if (faces.Get(i).cluster == cluster && faces.Get(i).Valid() && i != fstind)
 	    {
-	      const Point3d & p1 = points.Get(face.PNum(1)).P();
-	      const Point3d & p2 = points.Get(face.PNum(2)).P();
-	      const Point3d & p3 = points.Get(face.PNum(3)).P();
+	      const Point3d & p1 = points[face.PNum(1)].P();
+	      const Point3d & p2 = points[face.PNum(2)].P();
+	      const Point3d & p3 = points[face.PNum(3)].P();
 	      
 	      Box3d b2;
 	      b2.SetPoint (p1);
@@ -571,9 +563,9 @@ int AdFront3 :: GetLocals (int fstind,
   for (i = 1; i <= locfaces2.Size(); i++)
     {
       const Element2d & face = locfaces2.Get(i);
-      const Point3d & p1 = points.Get(face.PNum(1)).P();
-      const Point3d & p2 = points.Get(face.PNum(2)).P();
-      const Point3d & p3 = points.Get(face.PNum(3)).P();
+      const Point3d & p1 = points[face.PNum(1)].P();
+      const Point3d & p2 = points[face.PNum(2)].P();
+      const Point3d & p3 = points[face.PNum(3)].P();
 
       midp = Center (p1, p2, p3);
 
@@ -602,26 +594,22 @@ int AdFront3 :: GetLocals (int fstind,
     for (j = 1; j <= locfaces.Get(i).GetNP(); j++)
       {
 	pi = locfaces.Get(i).PNum(j);
-	invpindex.Elem(pi) = 0;
+	invpindex[pi] = -1;
       }
 
-  /*
-  for (i = 1; i <= points.Size(); i++)
-    invpindex.Elem(i) = 0;
-  */
   for (i = 1; i <= locfaces.Size(); i++)
     {
       for (j = 1; j <= locfaces.Get(i).GetNP(); j++)
 	{
 	  pi = locfaces.Get(i).PNum(j);
-	  if (invpindex.Get(pi) == 0)
+	  if (invpindex[pi] == -1)
 	    {
 	      pindex.Append (pi);
-	      invpindex.Elem(pi) = pindex.Size();
-	      locfaces.Elem(i).PNum(j) = locpoints.Append (points.Get(pi).P());
+	      invpindex[pi] = pindex.Size();  // -1+PointIndex::BASE;
+	      locfaces.Elem(i).PNum(j) = locpoints.Append (points[pi].P());
 	    }
 	  else
-	    locfaces.Elem(i).PNum(j) = invpindex.Get(pi);
+	    locfaces.Elem(i).PNum(j) = invpindex[pi];
 
 	}
     }
@@ -775,7 +763,7 @@ void AdFront3 :: SetStartFront (int /* baseelnp */)
       {
 	const Element2d & face = faces.Get(i).Face();
 	for (j = 1; j <= 3; j++)
-	  points.Elem(face.PNum(j)).DecFrontNr(0);
+	  points[face.PNum(j)].DecFrontNr(0);
       }
 
   /*
diff --git a/Netgen/libsrc/meshing/adfront3.hpp b/Netgen/libsrc/meshing/adfront3.hpp
index 50e30c56cb..2c43f9334b 100644
--- a/Netgen/libsrc/meshing/adfront3.hpp
+++ b/Netgen/libsrc/meshing/adfront3.hpp
@@ -110,23 +110,24 @@ public:
     }
   
   ///
-  int Valid () const
-  { 
-    return !f.IsDeleted(); // PNum(1) != 0; 
-  }
+  bool Valid () const
+  { return !f.IsDeleted(); }
 
   ///
   void Invalidate ();
+
   ///
-  int HashValue() const {return hashvalue;}
+  int HashValue() const 
+  { return hashvalue; }
+
   ///
-  void SetHashValue(int hv) {hashvalue = hv;}
+  void SetHashValue(int hv) 
+  { hashvalue = hv; }
 
   ///
   friend class AdFront3;
 
   int Cluster () const { return cluster; }
-      
 };  
 
 
@@ -136,14 +137,14 @@ public:
 class AdFront3
 {
   ///
-  ARRAY<FrontPoint3,PointIndex::BASE> points;
+  ARRAY<FrontPoint3, PointIndex::BASE> points;
   ///
   ARRAY<FrontFace> faces;
   ///
   ARRAY<PointIndex> delpointl;
 
   /// which points are connected to pi ?
-  TABLE<int> * connectedpairs;
+  TABLE<int, PointIndex::BASE> * connectedpairs;
   
   /// number of total front faces;
   int nff;
@@ -194,10 +195,10 @@ public:
   ///
   void Print () const;
   ///
-  int Empty () const
+  bool Empty () const
     { return nff == 0; }
   ///
-  int Empty (int elnp) const
+  bool Empty (int elnp) const
     {
       if (elnp == 4)
 	return (nff4 == 0);
@@ -238,7 +239,7 @@ public:
   ///
   void DeleteFace (INDEX fi);
   ///
-  INDEX AddPoint (const Point3d & p, PointIndex globind);
+  PointIndex AddPoint (const Point3d & p, PointIndex globind);
   ///
   INDEX AddFace (const Element2d & e);
   ///
@@ -253,7 +254,8 @@ public:
   /// is Point p inside Surface ?
   int Inside (const Point3d & p) const;
   /// both points on same side ?
-  int SameSide (const Point3d & lp1, const Point3d & lp2, const ARRAY<int> * testfaces = NULL) const;
+  int SameSide (const Point3d & lp1, const Point3d & lp2, 
+		const ARRAY<int> * testfaces = NULL) const;
 
 
   ///
diff --git a/Netgen/libsrc/meshing/bisect.cpp b/Netgen/libsrc/meshing/bisect.cpp
index 39420983c8..45d72b50d4 100644
--- a/Netgen/libsrc/meshing/bisect.cpp
+++ b/Netgen/libsrc/meshing/bisect.cpp
@@ -4,7 +4,7 @@
 
 namespace netgen
 {
-#include "../interface/writeuser.hpp"
+//#include "../interface/writeuser.hpp"
 
   class MarkedTet;
   class MarkedPrism;
@@ -1491,6 +1491,49 @@ void Refinement :: Bisect (Mesh & mesh,
   */
 
 
+
+  if (opt.refine_p)
+    {
+      int ne = mesh.GetNE();
+      int nse = mesh.GetNSE();
+      int ii = 0;
+      for (ElementIndex ei = 0; ei < ne; ei++)
+        if (mesh[ei].TestRefinementFlag())
+	  mesh[ei].SetOrder (mesh[ei].GetOrder()+1);
+
+      for (SurfaceElementIndex sei = 0; sei < nse; sei++)
+        if (mesh[sei].TestRefinementFlag())
+	  mesh[sei].SetOrder (mesh[sei].GetOrder()+1);
+
+
+      ARRAY<int,PointIndex::BASE> v_order (mesh.GetNP());
+      v_order = 0;
+
+      for (ElementIndex ei = 0; ei < ne; ei++)
+	for (j = 0; j < mesh[ei].GetNP(); j++)
+	  if (mesh[ei].GetOrder() > v_order[mesh[ei][j]])
+	    v_order[mesh[ei][j]] = mesh[ei].GetOrder();
+
+      for (SurfaceElementIndex sei = 0; sei < nse; sei++)
+	for (j = 0; j < mesh[sei].GetNP(); j++)
+	  if (mesh[sei].GetOrder() > v_order[mesh[sei][j]])
+	    v_order[mesh[sei][j]] = mesh[sei].GetOrder();
+
+      for (ElementIndex ei = 0; ei < ne; ei++)
+	for (j = 0; j < mesh[ei].GetNP(); j++)
+	  if (mesh[ei].GetOrder() < v_order[mesh[ei][j]]-1)
+	    mesh[ei].SetOrder(v_order[mesh[ei][j]]-1);
+
+      for (SurfaceElementIndex sei = 0; sei < nse; sei++)
+	for (j = 0; j < mesh[sei].GetNP(); j++)
+	  if (mesh[sei].GetOrder() < v_order[mesh[sei][j]]-1)
+	    mesh[sei].SetOrder(v_order[mesh[sei][j]]-1);
+
+      return;
+    }
+
+
+
   // INDEX_2_HASHTABLE<int> cutedges(10 + 5 * (mtets.Size()+mprisms.Size()+mtris.Size()+mquads.Size()));
   INDEX_2_CLOSED_HASHTABLE<int> cutedges(10 + 9 * (mtets.Size()+mprisms.Size()+mtris.Size()+mquads.Size()));
 
@@ -1526,7 +1569,7 @@ void Refinement :: Bisect (Mesh & mesh,
 		      mesh.VolumeElement(i).GetType() == TET10)
 		    {
 		      cnttet++;
-		      mtets.Elem(cnttet).marked = 
+		      mtets.Elem(cnttet).marked =
 			3 * mesh.VolumeElement(i).TestRefinementFlag();
 		      if (mtets.Elem(cnttet).marked)
 			cntm++;
@@ -1534,18 +1577,18 @@ void Refinement :: Bisect (Mesh & mesh,
 		  else
 		    {
 		      cntprism++;
-		      mprisms.Elem(cntprism).marked = 
+		      mprisms.Elem(cntprism).marked =
 			2 * mesh.VolumeElement(i).TestRefinementFlag();
 		      if (mprisms.Elem(cntprism).marked)
-			cntm++; 
+			cntm++;
 		    }
-		    
+
 		}
 	    }
 	  else
 	    for (i = 1; i <= mtets.Size(); i++)
 	      {
-		mtets.Elem(i).marked = 
+		mtets.Elem(i).marked =
 		  3 * mesh.VolumeElement(i).TestRefinementFlag();
 		if (mtets.Elem(i).marked)
 		  cntm++;
@@ -1570,8 +1613,8 @@ void Refinement :: Bisect (Mesh & mesh,
 		  mesh.SurfaceElement(i).GetType() == TRIG6)
 		{
 		  cnttrig++;
-		  mtris.Elem(cnttrig).marked = 
-		    2*mesh.SurfaceElement(i).TestRefinementFlag();
+		  mtris.Elem(cnttrig).marked =
+		    mesh.SurfaceElement(i).TestRefinementFlag() ? 2 : 0;
 		  // mtris.Elem(cnttrig).marked = 0;
 		  if (mtris.Elem(cnttrig).marked)
 		    cntm++;
@@ -1579,7 +1622,7 @@ void Refinement :: Bisect (Mesh & mesh,
 	      else
 		{
 		  cntquad++;
-		  mquads.Elem(cntquad).marked = 
+		  mquads.Elem(cntquad).marked =
 		    mesh.SurfaceElement(i).TestRefinementFlag();
 		  // mquads.Elem(cntquad).marked = 0;
 		  if (mquads.Elem(cntquad).marked)
@@ -1594,13 +1637,13 @@ void Refinement :: Bisect (Mesh & mesh,
 	      cntm = 0;
 	      for (i = 1; i <= mtris.Size(); i++)
 		{
-		  mtris.Elem(i).marked = 
+		  mtris.Elem(i).marked =
 		    2 * mesh.SurfaceElement(i).TestRefinementFlag();
 		  //		  mtris.Elem(i).marked = 2;
 		  if (mtris.Elem(i).marked)
 		    cntm++;
 		}
-	      
+
 	      if (!cntm)
 		{
 		  for (i = 1; i <= mtris.Size(); i++)
@@ -1623,13 +1666,42 @@ void Refinement :: Bisect (Mesh & mesh,
       if (!marked) break;
 
 
+	  if (opt.refine_p)
+	    {
+	      cout << "refine p" << endl;
+
+	      for (i = 1; i <= mtets.Size(); i++)
+		mtets.Elem(i).incorder = mtets.Elem(i).marked ? 1 : 0;
+
+	      for (i = 1; i <= mtets.Size(); i++)
+		if (mtets.Elem(i).incorder)
+		  mtets.Elem(i).marked = 0;
+
+
+	      for (i = 1; i <= mprisms.Size(); i++)
+		mprisms.Elem(i).incorder = mprisms.Elem(i).marked ? 1 : 0;
+
+	      for (i = 1; i <= mprisms.Size(); i++)
+		if (mprisms.Elem(i).incorder)
+		  mprisms.Elem(i).marked = 0;
+
+
+	      for (i = 1; i <= mtris.Size(); i++)
+		mtris.Elem(i).incorder = mtris.Elem(i).marked ? 1 : 0;
+
+	      for (i = 1; i <= mtris.Size(); i++)
+		{
+		  if (mtris.Elem(i).incorder)
+		    mtris.Elem(i).marked = 0;
+		}
+	    }
 
 	  if (opt.refine_hp)
 	    {
 	      cout << "refine hp" << endl;
 	      BitArray singv(np);
 	      singv.Clear();
-	      
+
 	      if (mesh.GetDimension() == 3)
 		for (i = 1; i <= mesh.GetNSeg(); i++)
 		  {
@@ -1655,7 +1727,7 @@ void Refinement :: Bisect (Mesh & mesh,
 			}
 		    }
 		}
-		
+
 
 
 	      for (i = 1; i <= mtets.Size(); i++)
@@ -2267,6 +2339,7 @@ BisectionOptions :: BisectionOptions ()
   maxlevel = 50;
   usemarkedelements = 0;
   refine_hp = 0;
+  refine_p = 0;
 }
 
 
diff --git a/Netgen/libsrc/meshing/bisect.hpp b/Netgen/libsrc/meshing/bisect.hpp
index 2d8988de0e..8a1e836116 100644
--- a/Netgen/libsrc/meshing/bisect.hpp
+++ b/Netgen/libsrc/meshing/bisect.hpp
@@ -11,6 +11,7 @@ public:
   int maxlevel;
   int usemarkedelements;
   bool refine_hp;
+  bool refine_p;
   BisectionOptions ();
 };
 
@@ -43,7 +44,7 @@ public:
   Refinement ();
   virtual ~Refinement ();
   
-  void Refine (Mesh & mesh, int levels);
+  void Refine (Mesh & mesh);
   void Bisect (Mesh & mesh, class BisectionOptions & opt);
   void MakeSecondOrder (Mesh & mesh);
 
@@ -61,6 +62,11 @@ public:
 
   virtual void ProjectToSurface (Point<3> & p, int surfi);
 
+  virtual void ProjectToSurface (Point<3> & p, int surfi, PointGeomInfo & /* gi */)
+  {
+    ProjectToSurface (p, surfi);
+  }
+
 
   void ValidateSecondOrder (Mesh & mesh);
   void ValidateRefinedMesh (Mesh & mesh, 
diff --git a/Netgen/libsrc/meshing/clusters.cpp b/Netgen/libsrc/meshing/clusters.cpp
index ed76676746..5eef2078e7 100644
--- a/Netgen/libsrc/meshing/clusters.cpp
+++ b/Netgen/libsrc/meshing/clusters.cpp
@@ -91,9 +91,13 @@ void AnisotropicClusters ::  Update()
 	cluster_reps.Elem(nnums[j]) = nnums[j];
     }
 
-
-
-
+  static const int hex_cluster[] =
+    { 
+      1, 2, 3, 4, 1, 2, 3, 4, 
+      5, 6, 7, 8, 5, 6, 7, 8, 1, 2, 3, 4, 
+      9, 9, 5, 8, 6, 7, 
+      9
+    };
 
   static const int prism_cluster[] =
     { 
@@ -162,6 +166,9 @@ void AnisotropicClusters ::  Update()
 	    case PRISM12:
 	      clustertab = prism_cluster;
 	      break;
+	    case HEX: 
+	      clustertab = hex_cluster; 
+	      break; 
 	    case PYRAMID:
 	      clustertab = pyramid_cluster;
 	      break;
diff --git a/Netgen/libsrc/meshing/curvedelems.cpp b/Netgen/libsrc/meshing/curvedelems.cpp
index fc0e590366..bce489fc84 100644
--- a/Netgen/libsrc/meshing/curvedelems.cpp
+++ b/Netgen/libsrc/meshing/curvedelems.cpp
@@ -132,18 +132,24 @@ namespace netgen
 	vdshape[0] = 1;
 	vdshape[1] = -1;
 
+	/*
 	if (edgeorient == -1)
 	{
 	    Swap (vshape[0], vshape[1]);
 	    Swap (vdshape[0], vdshape[1]);
 	}
+	*/
+	
     }
 
 
     void BaseFiniteElement1D :: CalcEdgeShapes ()
     {
 	b.SetOrder (edgeorder);
-	b.CalcFDf( 1-xi(0) );
+	if (edgeorient == 1)
+	  b.CalcFDf( 1-xi(0) );
+	else
+	  b.CalcFDf( xi(0) );
 
 	for (int k = 2; k <= edgeorder; k++)
 	{
@@ -156,7 +162,10 @@ namespace netgen
     void BaseFiniteElement1D :: CalcEdgeLaplaceShapes ()
     {
         b.SetOrder (edgeorder);
-	b.CalcDDf( 1-xi(0) );
+	if (edgeorient == 1)
+	  b.CalcDDf( 1-xi(0) );
+	else
+	  b.CalcDDf( xi(0) );
 
 	for (int k = 2; k <= edgeorder; k++)
 	    eddshape[k-2] = b.GetDDf(k);
@@ -175,7 +184,7 @@ namespace netgen
 	int locmaxedgeorder = -1;
 	
 	BaseFiniteElement<2> :: SetElementNumber (aelnr);
-	Element2d elem = mesh[(SurfaceElementIndex) (elnr-1)]; 
+	const Element2d & elem = mesh[(SurfaceElementIndex) (elnr-1)]; 
 	top.GetSurfaceElementEdges (elnr, &(edgenr[0]), &(edgeorient[0]));
 	facenr = top.GetSurfaceElementFace (elnr);
 	faceorient = top.GetSurfaceElementFaceOrientation (elnr);
@@ -233,19 +242,19 @@ namespace netgen
 	for (f = 0; f < nfaces; f++)
 	{
 	    surfacenr[f] = top.GetFace2SurfaceElement (facenr[f]);
-	    surfaceorient[f] = top.GetSurfaceElementFaceOrientation (surfacenr[f]);
+	    // surfaceorient[f] = top.GetSurfaceElementFaceOrientation (surfacenr[f]);
 	}
 	
 	for (e = 0; e < nedges; e++)
 	{
 	    edgeorder[e] = curv.GetEdgeOrder (edgenr[e]-1); // 1-based
-	    locmaxedgeorder = max2 (edgeorder[e], locmaxedgeorder);
+	    locmaxedgeorder = max (edgeorder[e], locmaxedgeorder);
 	}
 	
 	for (f = 0; f < nfaces; f++)
 	{
 	    faceorder[f] = curv.GetFaceOrder (facenr[f]-1); // 1-based
-	    locmaxfaceorder = max2 (faceorder[f], locmaxfaceorder);
+	    locmaxfaceorder = max (faceorder[f], locmaxfaceorder);
 	}
 	
 	CalcNFaceShapes ();
@@ -333,6 +342,8 @@ namespace netgen
 	int index = 0;
 	for (int e = 0; e < nedges; e++)
 	{
+	  if (edgeorder[e] <= 1) continue;
+
 	    int i0 = eledge[e][0]-1;
 	    int i1 = eledge[e][1]-1;
 
@@ -382,6 +393,8 @@ namespace netgen
 	    }
 	}
 	// (*testout) << "eshape = " << eshape << ", edshape = " << edshape << endl;
+
+
 	/*
 	index = 0;
 	for (int e = 0; e < nedges; e++)
@@ -916,6 +929,8 @@ namespace netgen
 
 	for (int f = 0; f < nfaces; f++)
 	{
+	  if (faceorder[f] <= 2) continue;
+
 	    int i0 = elface[f][0]-1;
 	    int i1 = elface[f][1]-1;
 	    int i2 = elface[f][2]-1;
@@ -1562,6 +1577,17 @@ namespace netgen
 
   int CurvedElements :: IsSurfaceElementCurved (int elnr) const
   {
+    if (mesh.coarsemesh)
+      {
+	const HPRefElement & hpref_el =
+	  (*mesh.hpelements) [ mesh[(SurfaceElementIndex) elnr].hp_elnr];
+
+	return mesh.coarsemesh->GetCurvedElements().IsSurfaceElementCurved (hpref_el.coarse_elnr);
+      }
+
+
+
+
     Element2d elem = mesh[(SurfaceElementIndex) elnr];
 
     switch (elem.GetType())
@@ -1590,6 +1616,16 @@ namespace netgen
 
   int CurvedElements :: IsElementCurved (int elnr) const
   {
+    if (mesh.coarsemesh)
+      {
+	const HPRefElement & hpref_el =
+	  (*mesh.hpelements) [ mesh[(ElementIndex) elnr].hp_elnr];
+
+	return mesh.coarsemesh->GetCurvedElements().IsElementCurved (hpref_el.coarse_elnr);
+      }
+
+
+
     Element elem = mesh[(ElementIndex) elnr];
 
     switch (elem.GetType())
@@ -1682,9 +1718,50 @@ namespace netgen
     void CurvedElements :: CalcSurfaceTransformation (Point<2> xi, int elnr,
 						      Point<3> * x, Mat<3,2> * dxdxi)
     {
-	Element2d elem = mesh[(SurfaceElementIndex) elnr];
 
-	BaseFiniteElement2D * fe2d;
+      if (mesh.coarsemesh)
+	{
+	  const HPRefElement & hpref_el =
+	    (*mesh.hpelements) [ mesh[(SurfaceElementIndex) elnr].hp_elnr];
+	  
+	  // xi umrechnen
+	  double lami[4];
+	  switch (mesh[(SurfaceElementIndex) elnr].GetType())
+	    {
+	    case TRIG: 
+	      {
+		lami[0] = xi(0);
+		lami[1] = xi(1);
+		lami[2] = 1-xi(0)-xi(1);
+		lami[3] = 0;
+		break;
+	      }
+	    case QUAD: 
+	      {
+		lami[0] = (1-xi(0))*(1-xi(1));
+		lami[1] = xi(0) * (1-xi(1));
+		lami[2] = xi(0) * xi(1);
+		lami[3] = (1-xi(0))*xi(1);
+		break;
+	      }
+	    }
+	  Point<2> coarse_xi(0,0);
+	  for (int i = 0; i < 4; i++)
+	    {
+	      coarse_xi(0) += hpref_el.param[i][0] * lami[i];
+	      coarse_xi(1) += hpref_el.param[i][1] * lami[i];
+	    }
+	  mesh.coarsemesh->GetCurvedElements().CalcSurfaceTransformation (coarse_xi, hpref_el.coarse_elnr, x, dxdxi);
+
+	  return;
+	}
+
+
+
+
+      const Element2d & elem = mesh[(SurfaceElementIndex) elnr];
+
+      BaseFiniteElement2D * fe2d;
 
 	// char locmem[max2(sizeof(FEQuad), sizeof(FETrig))];
 	char locmemtrig[sizeof(FETrig)];
@@ -1798,8 +1875,58 @@ namespace netgen
 
 
     void CurvedElements :: CalcElementTransformation (Point<3> xi, int elnr,
-				    Point<3> * x, Mat<3,3> * dxdxi)
+						      Point<3> * x, Mat<3,3> * dxdxi)
     {
+
+     if (mesh.coarsemesh)
+	{
+	  const HPRefElement & hpref_el =
+	    (*mesh.hpelements) [ mesh[(ElementIndex) elnr].hp_elnr];
+	  
+
+	  double lami[8];
+	  FlatVector vlami(8, lami);
+	  vlami = 0;
+	  mesh[(ElementIndex) elnr] . GetShapeNew (xi, vlami);
+
+	  Point<3> coarse_xi(0,0,0);
+	  for (int i = 0; i < 8; i++)
+	    for (int l = 0; l < 3; l++)
+	      coarse_xi(l) += hpref_el.param[i][l] * lami[i];
+	  
+	  Mat<3,3> trans, dxdxic;
+	  if (dxdxi)
+	    {
+	      MatrixFixWidth<3> dlami(8);
+	      dlami = 0;
+	      mesh[(ElementIndex) elnr] . GetDShapeNew (xi, dlami);	  
+	      
+	      trans = 0;
+	      for (int k = 0; k < 3; k++)
+		for (int l = 0; l < 3; l++)
+		  {
+		    double sum = 0;
+		    for (int i = 0; i < 8; i++)
+		      sum += hpref_el.param[i][l] * dlami(i, k);
+		    trans(l,k) = sum;
+		  }
+	    }
+	  
+	  mesh.coarsemesh->GetCurvedElements().CalcElementTransformation (coarse_xi, hpref_el.coarse_elnr, x, &dxdxic);
+
+	  if (dxdxi)
+	    *dxdxi = dxdxic * trans;
+	  return;
+	}
+
+
+
+
+
+
+
+
+
 	Element elem = mesh[(ElementIndex) elnr];
 	BaseFiniteElement3D * fe3d;
 
diff --git a/Netgen/libsrc/meshing/curvedelems.hpp b/Netgen/libsrc/meshing/curvedelems.hpp
index 7328d27d66..dd11eef051 100644
--- a/Netgen/libsrc/meshing/curvedelems.hpp
+++ b/Netgen/libsrc/meshing/curvedelems.hpp
@@ -78,6 +78,8 @@ public:
 
   bool IsHighOrder() const
   { return isHighOrder; };
+  void SetHighOrder () { isHighOrder = 1; }
+
 
   int GetNVisualSubsecs() const
   { return nvisualsubsecs; };
@@ -547,7 +549,7 @@ public:
     virtual ~FEQuad()
 	{};
 
-  virtual void SetVertexSingularity (int v, int exponent)
+  virtual void SetVertexSingularity (int /* v */, int /* exponent */)
 	{};
 
   virtual void CalcVertexShapes();
@@ -580,7 +582,7 @@ protected:
   int faceorient[6];
   int faceorder[6];
   int surfacenr[6];
-  int surfaceorient[6];
+  // int surfaceorient[6];
 
   int nfaceshapes[6];
 
diff --git a/Netgen/libsrc/meshing/curvedelems2.cpp b/Netgen/libsrc/meshing/curvedelems2.cpp
index d01a13af79..6b79ee5fe9 100644
--- a/Netgen/libsrc/meshing/curvedelems2.cpp
+++ b/Netgen/libsrc/meshing/curvedelems2.cpp
@@ -27,6 +27,15 @@ namespace netgen
 
     void CurvedElements :: BuildCurvedElements(Refinement * ref, int polydeg)
     {
+      if (mesh.coarsemesh)
+	{
+	  mesh.coarsemesh->GetCurvedElements().BuildCurvedElements (ref, polydeg);
+	  SetHighOrder();
+	  return;
+	}
+
+      PrintMessage (2, "Build curved elements, order = ", polydeg);
+
       NgLock lock(const_cast<Mesh&>(mesh).Mutex(), 1);
       isHighOrder = 0;
       lock.UnLock();
@@ -51,7 +60,7 @@ namespace netgen
 	edgeorder.SetSize (top.GetNEdges());
 	faceorder.SetSize (top.GetNFaces());
 
-	int nedgestocurve = top.GetNEdges();
+	int nedgestocurve = mesh.GetNSeg();
 
 	edgedone = 0;
 	edgeorder = 1;
@@ -174,13 +183,12 @@ namespace netgen
 		segm.CalcEdgeLaplaceShapes ();
 		
 		Point<3> xv(0,0,0);
+
 		for (int v = 0; v < 2; v++)
 		  xv = xv + segm.GetVertexShape(v) * mesh.Point(segm.GetVertexNr(v));
-		
+		  		
 		double secpoint = xi[l];
 
-		if (segm.GetEdgeOrientation() == -1) secpoint = 1. - secpoint; // reverse orientation
-		
 		ref->PointBetween (mesh.Point(segm.GetVertexNr(1)),
 				   mesh.Point(segm.GetVertexNr(0)), secpoint,
 				   s.surfnr2, s.surfnr1,
@@ -207,11 +215,12 @@ namespace netgen
 	
 	if (mesh.GetDimension() == 3)
 	  {
+	    int nedgescurved = mesh.GetNSeg();
 	    for (int i=0; i<mesh.GetNSE(); i++) 
 	      {
 		if (multithread.terminate) return;
 		
-		SetThreadPercent( double(100*(mesh.GetNSeg()+i)/nedgestocurve) );
+		//		SetThreadPercent( double(100*(mesh.GetNSeg()+i)/nedgestocurve) );
 		Element2d elem = mesh[(SurfaceElementIndex) i];
 		const ELEMENT_EDGE * eledges = MeshTopology::GetEdges(elem.GetType());
 		
@@ -219,13 +228,19 @@ namespace netgen
 		ARRAY<int> orient;
 		top.GetSurfaceElementEdges(i+1, edgenrs);
 		top.GetSurfaceElementEdgeOrientations(i+1, orient);
-		
+
 		for (int e = 0; e < top.GetNEdges(elem.GetType()); e++)
 		  {
+//		    cout << "e = " << e << "/" << top.GetNEdges(elem.GetType()) <<  endl;
+
+		    nedgescurved++;
+
 		    if (edgedone[edgenrs[e]-1]) continue;
 		    
 		    edgedone[edgenrs[e]-1] = 1;
 
+		    SetThreadPercent( double(100*(nedgescurved)/nedgestocurve) );
+
 		    edge.SetElementNumber (edgenrs[e]);
 
 		    for (k = 2; k <= edge.GetEdgeOrder(); k++)
@@ -233,6 +248,7 @@ namespace netgen
 
 		    for (int l = 0; l < nIntegrationPoints; l++)
 		      {
+//			cout << "." << flush;
 			edge.SetReferencePoint (Point<1>(xi[l]));
 			edge.CalcVertexShapes ();
 			edge.CalcEdgeLaplaceShapes ();
@@ -243,18 +259,26 @@ namespace netgen
 
 			double secpoint = xi[l];
 
-			ref->PointBetween (mesh.Point(edge.GetVertexNr(1)),
-					   mesh.Point(edge.GetVertexNr(0)), secpoint,
-					   mesh.GetFaceDescriptor(elem.GetIndex()).SurfNr(),
-					   elem.GeomInfoPi(eledges[e][1]),
-					   elem.GeomInfoPi(eledges[e][0]),
-					   xexact, newgi);
+			if (orient[e] == 1)
+			  ref->PointBetween (mesh.Point(edge.GetVertexNr(1)),
+					     mesh.Point(edge.GetVertexNr(0)), secpoint,
+					     mesh.GetFaceDescriptor(elem.GetIndex()).SurfNr(),
+					     elem.GeomInfoPi(eledges[e][1]),
+					     elem.GeomInfoPi(eledges[e][0]),
+					     xexact, newgi);
+			else
+			  ref->PointBetween (mesh.Point(edge.GetVertexNr(1)),
+					     mesh.Point(edge.GetVertexNr(0)), secpoint,
+					     mesh.GetFaceDescriptor(elem.GetIndex()).SurfNr(),
+					     elem.GeomInfoPi(eledges[e][0]),
+					     elem.GeomInfoPi(eledges[e][1]),
+					     xexact, newgi);
 
 			for (k = 2; k <= edge.GetEdgeOrder(); k++)
 			  edgecoeffs[edgecoeffsindex[edgenrs[e]-1]+k-2] -=
 			    wi[l] * edge.GetEdgeLaplaceShape(k-2) * Vec<3>(xexact - xv);
 		      }	
-
+//		    cout << endl;
 		    for (k = 2; k <= edge.GetEdgeOrder(); k++)
 		      edgecoeffs[edgecoeffsindex[edgenrs[e]-1]+k-2] =
 			(2.0*(k-1.0)+1.0)*edgecoeffs[edgecoeffsindex[edgenrs[e]-1]+k-2];
@@ -512,11 +536,20 @@ namespace netgen
 			// integration over the difference between the exact geometry and the one
 			// defined by vertex and edge shape functions times face shape
 
+			// double giu = 0, giv = 0;
+			PointGeomInfo gi;
+			gi.trignum = elem.GeomInfoPi(1).trignum;
+			gi.u = 0.0;
+			gi.v = 0.0;
 			Point<3> xve(0.,0.,0.);
 
 			// vertex shape functions
 			for (int v = 0; v < fe2d->GetNVertices(); v++)
-			  xve = xve + fe2d->GetVertexShape(v) * mesh.Point(fe2d->GetVertexNr(v));
+			  {
+			    xve = xve + fe2d->GetVertexShape(v) * mesh.Point(fe2d->GetVertexNr(v));
+			    gi.u += fe2d->GetVertexShape(v) * elem.GeomInfoPi(v+1).u;
+			    gi.v += fe2d->GetVertexShape(v) * elem.GeomInfoPi(v+1).v;
+			  }
 
 			// edge shape functions
 			int index = 0;
@@ -530,7 +563,7 @@ namespace netgen
 			// exact point
 
 			Point<3> xexact = xve;
-			ref->ProjectToSurface (xexact, mesh.GetFaceDescriptor(elem.GetIndex()).SurfNr());
+			ref->ProjectToSurface (xexact, mesh.GetFaceDescriptor(elem.GetIndex()).SurfNr(), gi);
 
 			Vec<3> v2 = w*(Vec<3>(xexact)-Vec<3>(xve));
 
@@ -704,7 +737,7 @@ namespace netgen
 	  double maxcoeff = 0.;
 
 	  for (i = facecoeffsindex[f]; i < facecoeffsindex[f+1]; i++)
-	    maxcoeff = max2 (maxcoeff, facecoeffs[i].Length());
+	    maxcoeff = max (maxcoeff, facecoeffs[i].Length());
 
 	  if (maxcoeff < 1e-12) faceorder[f] = 1;
 	}
diff --git a/Netgen/libsrc/meshing/delaunay.cpp b/Netgen/libsrc/meshing/delaunay.cpp
index 8e44c5f739..505a8c193f 100644
--- a/Netgen/libsrc/meshing/delaunay.cpp
+++ b/Netgen/libsrc/meshing/delaunay.cpp
@@ -2,11 +2,18 @@
 #include "meshing.hpp"
 
 
-// #define TEST 
-// #define TEST2
 
 namespace netgen
 {
+
+
+  static const int deltetfaces[][3] = 
+    { { 1, 2, 3 },
+      { 2, 0, 3 },
+      { 0, 1, 3 },
+      { 1, 0, 2 } };
+
+
   class DelaunayTet
   {
     PointIndex pnums[4];
@@ -27,53 +34,62 @@ namespace netgen
 	pnums[i] = el[i];
     }
     
-    PointIndex & PNum(int i) { return pnums[i-1]; }
-    PointIndex PNum(int i) const { return pnums[i-1]; }
-
     PointIndex & operator[] (int i) { return pnums[i]; }
     PointIndex operator[] (int i) const { return pnums[i]; }
 
-    int & NB(int i) { return nb[i-1]; }
-    int NB(int i) const { return nb[i-1]; }
-
-    inline void GetFace (int i, INDEX_3 & face) const;
-    inline int FaceNr (INDEX_3 & face) const;  // which face nr is it ?
-    inline void GetFace (int i, Element2d & face) const;
-  };
+    int & NB1(int i) { return nb[i-1]; }
+    int NB1(int i) const { return nb[i-1]; }
 
+    int & NB(int i) { return nb[i]; }
+    int NB(int i) const { return nb[i]; }
 
 
-  inline int DelaunayTet :: FaceNr (INDEX_3 & face) const
-  {
-    for (int i = 0; i < 3; i++)
-      if (pnums[i] != face.I1() && 
-	  pnums[i] != face.I2() && 
-	  pnums[i] != face.I3())
-	return i+1;
-    return 4;
-  }
-
-  static const int deltetfaces[][3] = 
-    { { 2, 3, 4 },
-      { 3, 1, 4 },
-      { 1, 2, 4 },
-      { 2, 1, 3 } };
-
+    int FaceNr (INDEX_3 & face) const  // which face nr is it ?
+    {
+      for (int i = 0; i < 3; i++)
+	if (pnums[i] != face.I1() && 
+	    pnums[i] != face.I2() && 
+	    pnums[i] != face.I3())
+	  return i;
+      return 3;
+    }
+    
+    void GetFace1 (int i, INDEX_3 & face) const
+    {
+      face.I(1) = pnums[deltetfaces[i-1][0]];
+      face.I(2) = pnums[deltetfaces[i-1][1]];
+      face.I(3) = pnums[deltetfaces[i-1][2]];
+    }
 
-  inline void DelaunayTet :: GetFace (int i, INDEX_3 & face) const
-  {
-    face.I(1) = PNum(deltetfaces[i-1][0]);
-    face.I(2) = PNum(deltetfaces[i-1][1]);
-    face.I(3) = PNum(deltetfaces[i-1][2]);
-  }
+    void GetFace (int i, INDEX_3 & face) const
+    {
+      face.I(1) = pnums[deltetfaces[i][0]];
+      face.I(2) = pnums[deltetfaces[i][1]];
+      face.I(3) = pnums[deltetfaces[i][2]];
+    }
+      
+    INDEX_3 GetFace1 (int i) const
+    {
+      return INDEX_3 (pnums[deltetfaces[i-1][0]],
+		      pnums[deltetfaces[i-1][1]],
+		      pnums[deltetfaces[i-1][2]]);
+    }
 
-  inline void DelaunayTet :: GetFace (int i, Element2d & face) const
-  {
-    face.SetType(TRIG);
-    face[0] = PNum(deltetfaces[i-1][0]);
-    face[1] = PNum(deltetfaces[i-1][1]);
-    face[2] = PNum(deltetfaces[i-1][2]);
-  }
+    INDEX_3 GetFace (int i) const
+    {
+      return INDEX_3 (pnums[deltetfaces[i][0]],
+		      pnums[deltetfaces[i][1]],
+		      pnums[deltetfaces[i][2]]);
+    }
+     
+    void GetFace1 (int i, Element2d & face) const
+    {
+      // face.SetType(TRIG);
+      face[0] = pnums[deltetfaces[i-1][0]];
+      face[1] = pnums[deltetfaces[i-1][1]];
+      face[2] = pnums[deltetfaces[i-1][2]];
+    }
+  };
 
 
 
@@ -90,8 +106,10 @@ namespace netgen
   {
     // face nodes -> one element
     INDEX_3_CLOSED_HASHTABLE<int> faces;
-    //
+
+    // 
     ARRAY<DelaunayTet> & tets;
+
   public:
 
     // estimated number of points
@@ -99,48 +117,48 @@ namespace netgen
       : faces(200), tets(atets)
     { ; }
 
-
     // add element with 4 nodes
     void Add (int elnr);
 
     // delete element with 4 nodes
-    void Delete (int elnr);
+    void Delete (int elnr)
+    {
+      DelaunayTet & el = tets.Elem(elnr);
+      for (int i = 0; i < 4; i++)
+	faces.Set (el.GetFace(i).Sort(), el.NB(i));
+    }
 
     // get neighbour of element elnr in direction fnr 
     int GetNB (int elnr, int fnr)
-    { return tets.Get(elnr).NB(fnr); }
+    { 
+      return tets.Get(elnr).NB1(fnr); 
+    }
 
     //
     void ResetFaceHT (int size)
     {
       faces.SetSize (size);
     }
-    
-    void PrintMemInfo (ostream & ost) const;
   };
 
 
 
   void MeshNB :: Add (int elnr)
   {
-    INDEX_3 i3, i32; 
-
     DelaunayTet & el = tets.Elem(elnr);
 
-    for (int i = 1; i <= 4; i++)
+    for (int i = 0; i < 4; i++)
       {
-	el.GetFace (i, i3);
-	i3.Sort();
-      
-	int bnr, posnr, othertet;
+	INDEX_3 i3 = INDEX_3::Sort (el.GetFace(i));
 
+	int posnr;
+	
 	if (!faces.PositionCreate (i3, posnr))
 	  {
 	    // face already in use
-	    faces.GetData (posnr, othertet);
+	    int othertet = faces.GetData (posnr);
 
 	    el.NB(i) = othertet;
-	  
 	    if (othertet)
 	      {
 		int fnr = tets.Get(othertet).FaceNr (i3);
@@ -156,36 +174,6 @@ namespace netgen
   }
 
 
-  void MeshNB :: Delete (int elnr)
-  {
-    INDEX_3 i3; 
-    DelaunayTet & el = tets.Elem(elnr);
-
-    for (int i = 1; i <= 4; i++)
-      {
-	el.GetFace (i, i3);
-	i3.Sort();
-	faces.Set (i3, el.NB(i));
-      } 
-  }
-
-
-  void MeshNB :: PrintMemInfo (ostream & ost) const
-  {
-    /*
-      int uf = 0;
-      for (int i = 1; i <= face2el.Size(); i++)
-      if (face2el.Get(i).I1())
-      uf++;
-
-      ost << "MeshNB: "
-      << "validels = " << validels << " totels = " << el2face.Size() << endl
-      << "validfaces = " << uf << " totfaces = " << face2el.Size() << endl
-      << "face2el: " << face2el.Size() * sizeof(INDEX_2) << endl
-      << "el2face: " << el2face.Size() * sizeof(INDEX_4) << endl;
-    */
-  }
-
 
 
 
@@ -197,30 +185,30 @@ namespace netgen
   {
     ARRAY<int> links;
   public:
-    SphereList () { ; }
-    inline void AddElement (int elnr);
-    inline void DeleteElement (int elnr);
-    inline void ConnectElement (int eli, int toi);
-    void GetList (int eli, ARRAY<int> & linked) const;
-  };
+    SphereList () 
+    { ; }
 
-  inline void SphereList :: AddElement (int elnr)
-  {
-    if (elnr > links.Size())
-      links.Append (1);
-    links.Elem(elnr) = elnr;
-  }
+    void AddElement (int elnr)
+    {
+      if (elnr > links.Size())
+	links.Append (1);
+      links.Elem(elnr) = elnr;
+    }
 
-  inline void SphereList :: DeleteElement (int elnr)
-  {
-    links.Elem(elnr) = 0;
-  }
+    void DeleteElement (int elnr)
+    {
+      links.Elem(elnr) = 0;
+    }    
+  
+    void ConnectElement (int eli, int toi)
+    {
+      links.Elem (eli) = links.Get (toi);
+      links.Elem (toi) = eli;
+    }
+      
+    void GetList (int eli, ARRAY<int> & linked) const;
+  };
 
-  inline void SphereList :: ConnectElement (int eli, int toi)
-  {
-    links.Elem (eli) = links.Get (toi);
-    links.Elem (toi) = eli;
-  }
 
   void SphereList :: GetList (int eli, ARRAY<int> & linked) const
   {
@@ -262,10 +250,6 @@ namespace netgen
 			 IndexSet & insphere, IndexSet & closesphere)
   {
     int i, j, k, l;
-
-#ifdef TEST2
-    (*testout) << endl << "add point " << newp << endl;
-#endif
   
     /*
       find any sphere, such that newp is contained in
@@ -369,7 +353,7 @@ namespace netgen
 	      {
 		int celind = connected.Get(k);
 
-		if (tempels.Get(celind).PNum(1) != -1 && 
+		if (tempels.Get(celind)[0] != -1 && 
 		    !insphere.IsIn (celind))
 		  {
 		    changed = 1;
@@ -402,19 +386,28 @@ namespace netgen
 		    }
 		  else
 		    {
+		      /*
 		      Element2d face;
 		      tempels.Get(helind).GetFace (k, face);
 
 		      const Point3d & p1 = mesh.Point (face.PNum(1));
-		      const Point3d & p2 = mesh.Point (face.PNum(2));
-		      const Point3d & p3 = mesh.Point (face.PNum(3));
+		      const Point3d & p2 = mesh.Point (face[1]);
+		      const Point3d & p3 = mesh.Point (face[2]);
+		      */
+
+		      INDEX_3 i3 = tempels.Get(helind).GetFace (k-1);
+
+		      const Point3d & p1 = mesh.Point ( PointIndex (i3.I1()));
+		      const Point3d & p2 = mesh.Point ( PointIndex (i3.I2()));
+		      const Point3d & p3 = mesh.Point ( PointIndex (i3.I3()));
+
 
 		      Vec3d v1(p1, p2);
 		      Vec3d v2(p1, p3);
 		      Vec3d n = Cross (v1, v2);
 		      n /= n.Length();
 
-		      if (n * Vec3d (p1, mesh.Point (tempels.Get(helind).PNum(k))) > 0)
+		      if (n * Vec3d (p1, mesh.Point (tempels.Get(helind)[k-1])) > 0)
 			n *= -1;
 
 		      double dist = n * Vec3d (p1, newp);
@@ -435,6 +428,7 @@ namespace netgen
     //      (*testout) << "newels: " << endl;
     ARRAY<Element> newels;
 
+    Element2d face(TRIG);
     for (j = 1; j <= insphere.Array().Size(); j++)
       for (k = 1; k <= 4; k++)
 	{
@@ -444,26 +438,25 @@ namespace netgen
 
 	  if (!nbind || !insphere.IsIn (nbind))
 	    {
-	      Element2d face;
-	      tempels.Get (celind).GetFace (k, face);
+	      tempels.Get (celind).GetFace1 (k, face);
 		
-	      Element newel(4);
+	      Element newel(TET);
 	      for (l = 1; l <= 3; l++)
 		newel.PNum(l) = face.PNum(l);
-	      newel.PNum(4) = newpi;
+	      newel[3] = newpi;
 
 	      newels.Append (newel);
 
-	      Vec3d v1(mesh.Point (face.PNum(1)), mesh.Point (face.PNum(2)));
-	      Vec3d v2(mesh.Point (face.PNum(1)), mesh.Point (face.PNum(3)));
+	      Vec3d v1(mesh.Point (face[0]), mesh.Point (face[1]));
+	      Vec3d v2(mesh.Point (face[0]), mesh.Point (face[2]));
 	      Vec3d n = Cross (v1, v2);
 	      n /= n.Length();
-	      if (n * Vec3d(mesh.Point (face.PNum(1)), 
-			    mesh.Point (tempels.Get(insphere.Array().Get(j)).PNum(k)))
+	      if (n * Vec3d(mesh.Point (face[0]), 
+			    mesh.Point (tempels.Get(insphere.Array().Get(j))[k-1]))
 		  > 0)
 		n *= -1;
 
-	      double hval = n * Vec3d (mesh.Point (face.PNum(1)), newp);
+	      double hval = n * Vec3d (mesh.Point (face[0]), newp);
 		
 	      if (hval > -1e-12)
 		{
@@ -471,9 +464,9 @@ namespace netgen
 		  (*testout) << "vec to outer, hval = " << hval << endl;
 		  (*testout) << "v1 x v2 = " << Cross (v1, v2) << endl;
 		  (*testout) << "facep: "
-			     << mesh.Point (face.PNum(1)) << " "
-			     << mesh.Point (face.PNum(2)) << " "
-			     << mesh.Point (face.PNum(3)) << endl;
+			     << mesh.Point (face[0]) << " "
+			     << mesh.Point (face[1]) << " "
+			     << mesh.Point (face[2]) << endl;
 		}
 	    }
 	}
@@ -488,8 +481,8 @@ namespace netgen
 	meshnb.Delete (celind); 
 	list.DeleteElement (celind);
 	  
-	for (k = 1; k <= 4; k++)
-	  tempels.Elem(celind).PNum(k) = -1;
+	for (k = 0; k < 4; k++)
+	  tempels.Elem(celind)[k] = -1;
 
 	((ADTree6&)tettree.Tree()).DeleteElement (celind);
 	freelist.Append (celind);
@@ -600,7 +593,7 @@ namespace netgen
 
 
     // new: local box
-    mesh.GetBox (pmax, pmin);
+    mesh.GetBox (pmax, pmin);   // lower bound for pmax, upper for pmin
     for (i = 1; i <= adfront->GetNF(); i++)
       {
 	const Element2d & face = adfront->GetFace(i);
@@ -611,6 +604,13 @@ namespace netgen
 	  }
       }
   
+    for (i = 0; i < mesh.LockedPoints().Size(); i++)
+      {
+	pmin.SetToMin (mesh.Point (mesh.LockedPoints()[i]));
+	pmax.SetToMax (mesh.Point (mesh.LockedPoints()[i]));
+      }
+  
+
 
     Vec3d vdiag(pmin, pmax);
     // double r1 = vdiag.Length();
@@ -631,10 +631,10 @@ namespace netgen
 
     int np = mesh.GetNP();
 
-    startel.PNum(1) = mesh.AddPoint (cp1);
-    startel.PNum(2) = mesh.AddPoint (cp2);
-    startel.PNum(3) = mesh.AddPoint (cp3);
-    startel.PNum(4) = mesh.AddPoint (cp4);
+    startel[0] = mesh.AddPoint (cp1);
+    startel[1] = mesh.AddPoint (cp2);
+    startel[2] = mesh.AddPoint (cp3);
+    startel[3] = mesh.AddPoint (cp4);
 
     // flag points to use for Delaunay:
     BitArrayChar<PointIndex::BASE> usep(np);
@@ -674,11 +674,11 @@ namespace netgen
     ARRAY<int> connected, treesearch;
 
 
-    tpmin = tpmax = mesh.Point(startel.PNum(1));
-    for (k = 2; k <= 4; k++)
+    tpmin = tpmax = mesh.Point(startel[0]);
+    for (k = 1; k < 4; k++)
       {
-	tpmin.SetToMin (mesh.Point (startel.PNum(k)));
-	tpmax.SetToMax (mesh.Point (startel.PNum(k)));
+	tpmin.SetToMin (mesh.Point (startel[k]));
+	tpmax.SetToMax (mesh.Point (startel[k]));
       }
     tpmax = tpmax + 0.01 * (tpmax - tpmin);
     tettree.Insert (tpmin, tpmax, 1);
@@ -686,8 +686,8 @@ namespace netgen
 
     Point3d pc;
 	  
-    for (k = 1; k <= 4; k++)
-      pp[k-1] = &mesh.Point (startel.PNum(k));
+    for (k = 0; k < 4; k++)
+      pp[k] = &mesh.Point (startel[k]);
   
     CalcSphereCenter (&pp[0], pc);
 
@@ -748,7 +748,7 @@ namespace netgen
 
 
     for (i = tempels.Size(); i >= 1; i--)
-      if (tempels.Get(i).PNum(1) <= 0)
+      if (tempels.Get(i)[0] <= 0)
 	tempels.DeleteElement (i);
 
     PrintDot ('\n');
@@ -816,14 +816,15 @@ namespace netgen
       for (i = 1; i <= tempels.Size(); i++)
 	{   
 	  Element el(4);
-	  for (j = 1; j <= 4; j++)
-	    el.PNum(j) = tempels.Elem(i).PNum(j);
+	  for (j = 0; j < 4; j++)
+	    el[j] = tempels.Elem(i)[j];
 
 	  el.SetIndex (1);
-	  const Point3d & lp1 = mesh.Point (el.PNum(1));
-	  const Point3d & lp2 = mesh.Point (el.PNum(2));
-	  const Point3d & lp3 = mesh.Point (el.PNum(3));
-	  const Point3d & lp4 = mesh.Point (el.PNum(4));
+
+	  const Point3d & lp1 = mesh.Point (el[0]);
+	  const Point3d & lp2 = mesh.Point (el[1]);
+	  const Point3d & lp3 = mesh.Point (el[2]);
+	  const Point3d & lp4 = mesh.Point (el[3]);
 	  Vec3d v1(lp1, lp2);
 	  Vec3d v2(lp1, lp3);
 	  Vec3d v3(lp1, lp4);
@@ -831,11 +832,12 @@ namespace netgen
 	  double vol = n * v3;
 	
 	  if (vol > 0)
-	    Swap (el.PNum(3), el.PNum(4));
+	    swap (el[2], el[3]);
 	
 	  tempmesh.AddVolumeElement (el);
 	}
 
+
       MeshQuality3d (tempmesh);
     
       for (i = 1; i <= mesh.GetNOpenElements(); i++)
@@ -843,23 +845,23 @@ namespace netgen
 	  Element2d sel = mesh.OpenElement(i);
 	  sel.SetIndex(1);
 	  tempmesh.AddSurfaceElement (sel);
-	  Swap (sel.PNum(2), sel.PNum(3));
+	  swap (sel[1], sel[2]);
 	  tempmesh.AddSurfaceElement (sel);
 	}
 
 
       for (i = 1; i <= 4; i++)
 	{
-	  Element2d self(3);
+	  Element2d self(TRIG);
 	  self.SetIndex (1);
-	  startel.GetFace (i, self);
+	  startel.GetFace1 (i, self);
 	  tempmesh.AddSurfaceElement (self);
 	}
 
       
       tempmesh.AddFaceDescriptor (FaceDescriptor (1, 1, 0, 0));
       tempmesh.AddFaceDescriptor (FaceDescriptor (2, 1, 0, 0));
-    
+
 
       //  for (i = mesh.GetNP() - 3; i <= mesh.GetNP(); i++)
       //    tempmesh.AddLockedPoint (i);
@@ -899,7 +901,6 @@ namespace netgen
 
 
 
-
     // remove degenerated
 
     BitArray badnode(mesh.GetNP());
@@ -908,13 +909,13 @@ namespace netgen
     for (i = 1; i <= tempels.Size(); i++)
       {
 	Element el(4);
-	for (j = 1; j <= 4; j++)
-	  el.PNum(j) = tempels.Elem(i).PNum(j);
+	for (j = 0; j < 4; j++)
+	  el[j] = tempels.Elem(i)[j];
 	//      Element & el = tempels.Elem(i);
-	const Point3d & lp1 = mesh.Point (el.PNum(1));
-	const Point3d & lp2 = mesh.Point (el.PNum(2));
-	const Point3d & lp3 = mesh.Point (el.PNum(3));
-	const Point3d & lp4 = mesh.Point (el.PNum(4));
+	const Point3d & lp1 = mesh.Point (el[0]);
+	const Point3d & lp2 = mesh.Point (el[1]);
+	const Point3d & lp3 = mesh.Point (el[2]);
+	const Point3d & lp4 = mesh.Point (el[3]);
 	Vec3d v1(lp1, lp2);
 	Vec3d v2(lp1, lp3);
 	Vec3d v3(lp1, lp4);
@@ -923,29 +924,29 @@ namespace netgen
 
 	double h = v1.Length() + v2.Length() + v3.Length();
 	if (fabs (vol) < 1e-8 * (h * h * h) &&
-	    (el.PNum(1) <= np && el.PNum(2) <= np &&
-	     el.PNum(3) <= np && el.PNum(4) <= np) )   // old: 1e-12
+	    (el[0] <= np && el[1] <= np &&
+	     el[2] <= np && el[3] <= np) )   // old: 1e-12
 	  {
-	    badnode.Set(el.PNum(1));
-	    badnode.Set(el.PNum(2));
-	    badnode.Set(el.PNum(3));
-	    badnode.Set(el.PNum(4));
+	    badnode.Set(el[0]);
+	    badnode.Set(el[1]);
+	    badnode.Set(el[2]);
+	    badnode.Set(el[3]);
 	    ndeg++;
 	    (*testout) << "vol = " << vol << " h = " << h << endl;
 	  }
 
 	if (vol > 0)
-	  Swap (el.PNum(3), el.PNum(4));
+	  Swap (el[2], el[3]);
       }
 
     ne = tempels.Size();
     for (i = ne; i >= 1; i--)
       {
 	const DelaunayTet & el = tempels.Get(i);
-	if (badnode.Test(el.PNum(1)) ||
-	    badnode.Test(el.PNum(2)) ||
-	    badnode.Test(el.PNum(3)) ||
-	    badnode.Test(el.PNum(4)) )
+	if (badnode.Test(el[0]) ||
+	    badnode.Test(el[1]) ||
+	    badnode.Test(el[2]) ||
+	    badnode.Test(el[3]) )
 	  tempels.DeleteElement(i);
       }
 
@@ -959,18 +960,16 @@ namespace netgen
     for (i = 1; i <= mesh.GetNOpenElements(); i++)
       {
 	const Element2d & tri = mesh.OpenElement(i);
-	INDEX_3 i3(tri.PNum(1), tri.PNum(2), tri.PNum(3));
+	INDEX_3 i3(tri[0], tri[1], tri[2]);
 	i3.Sort();
 	openeltab.Set (i3, i);
       }
 
     for (i = 1; i <= tempels.Size(); i++)
       {
-	for (j = 1; j <= 4; j++)
+	for (j = 0; j < 4; j++)
 	  {
-	    Element2d face;
-	    tempels.Get(i).GetFace (j, face);
-	    INDEX_3 i3(face.PNum(1), face.PNum(2), face.PNum(3));
+	    INDEX_3 i3 = tempels.Get(i).GetFace (j);
 	    i3.Sort();
 	    if (openeltab.Used(i3))
 	      openeltab.Set (i3, 0);
@@ -1028,12 +1027,12 @@ namespace netgen
 	  {
 	    switch (j)
 	      {
-	      case 1: i2 = INDEX_2(el.PNum(1), el.PNum(2)); break;
-	      case 2: i2 = INDEX_2(el.PNum(1), el.PNum(3)); break;
-	      case 3: i2 = INDEX_2(el.PNum(1), el.PNum(4)); break;
-	      case 4: i2 = INDEX_2(el.PNum(2), el.PNum(3)); break;
-	      case 5: i2 = INDEX_2(el.PNum(2), el.PNum(4)); break;
-	      case 6: i2 = INDEX_2(el.PNum(3), el.PNum(4)); break;
+	      case 1: i2 = INDEX_2(el[0], el[1]); break;
+	      case 2: i2 = INDEX_2(el[0], el[2]); break;
+	      case 3: i2 = INDEX_2(el[0], el[3]); break;
+	      case 4: i2 = INDEX_2(el[1], el[2]); break;
+	      case 5: i2 = INDEX_2(el[1], el[3]); break;
+	      case 6: i2 = INDEX_2(el[2], el[3]); break;
 	      }
 	    i2.Sort();
 	    tetedges.Set (i2, 1);
@@ -1042,6 +1041,35 @@ namespace netgen
     //  cout << "tetedges:";
     //  tetedges.PrintMemInfo (cout);
 
+
+    for (INDEX_2_HASHTABLE<INDEX_2>::Iterator it = twotrias.Begin();
+	 it != twotrias.End(); it++)
+      {
+	INDEX_2 hi2, hi3;
+	twotrias.GetData (it, hi2, hi3);
+	hi3.Sort();
+	if (tetedges.Used (hi3))
+	  {
+	    const Point3d & p1 = mesh.Point ( PointIndex (hi2.I1()));
+	    const Point3d & p2 = mesh.Point ( PointIndex (hi2.I2()));
+	    const Point3d & p3 = mesh.Point ( PointIndex (hi3.I1()));
+	    const Point3d & p4 = mesh.Point ( PointIndex (hi3.I2()));
+	    Vec3d v1(p1, p2);
+	    Vec3d v2(p1, p3);
+	    Vec3d v3(p1, p4);
+	    Vec3d n = Cross (v1, v2);
+	    double vol = n * v3;
+	    
+	    double h = v1.Length() + v2.Length() + v3.Length();
+	    if (fabs (vol) < 1e-4 * (h * h * h))   // old: 1e-12
+	      {
+		badnode.Set(hi3.I1());	
+		badnode.Set(hi3.I2());	
+	      }
+	  }
+      }
+
+    /*
     for (i = 1; i <= twotrias.GetNBags(); i++)
       for (j = 1; j <= twotrias.GetBagSize (i); j++)
 	{
@@ -1068,16 +1096,16 @@ namespace netgen
 		}
 	    }
 	}
-
+    */
 
     ne = tempels.Size();
     for (i = ne; i >= 1; i--)
       {
 	const DelaunayTet & el = tempels.Get(i);
-	if (badnode.Test(el.PNum(1)) ||
-	    badnode.Test(el.PNum(2)) ||
-	    badnode.Test(el.PNum(3)) ||
-	    badnode.Test(el.PNum(4)) )
+	if (badnode.Test(el[0]) ||
+	    badnode.Test(el[1]) ||
+	    badnode.Test(el[2]) ||
+	    badnode.Test(el[3]) )
 	  tempels.DeleteElement(i);
       }
 
@@ -1103,7 +1131,7 @@ namespace netgen
 	      {
 		const Element2d & tri = mesh.OpenElement(fnr);
 	      
-		Point3d ltpmin (mesh.Point(tri.PNum(1)));
+		Point3d ltpmin (mesh.Point(tri[0]));
 		Point3d ltpmax (tpmin);
 	      
 		for (k = 2; k <= 3; k++)
@@ -1124,10 +1152,10 @@ namespace netgen
 	  
 	    int intersect = 0;
 	  
-	    for (j = 1; j <= 4; j++)
+	    for (j = 0; j < 4; j++)
 	      {
-		pp[j-1] = &mesh.Point(el.PNum(j));
-		tetpi[j-1] = el.PNum(j);
+		pp[j] = &mesh.Point(el[j]);
+		tetpi[j] = el[j];
 	      }
 	  
 	    Point3d tetpmin(*pp[0]);
@@ -1161,6 +1189,7 @@ namespace netgen
 	      
 		if (IntersectTetTriangle (&pp[0], &tripp[0], tetpi, tripi))
 		  {
+		    /*
 		    int il1, il2;
 		    (*testout) << "intersect !" << endl;
 		    (*testout) << "triind: ";
@@ -1180,7 +1209,7 @@ namespace netgen
 		    for (il2 = 0; il2 < 4; il2++)
 		      (*testout) << " " << *pp[il2];
 		    (*testout) << endl;
-	      
+		    */
 		  
 		  
 		    intersect = 1;
@@ -1220,7 +1249,7 @@ namespace netgen
     for (i = 1; i <= mesh.GetNOpenElements(); i++)
       {
 	const Element2d & tri = mesh.OpenElement(i);
-	INDEX_3 i3 (tri.PNum(1), tri.PNum(2), tri.PNum(3));
+	INDEX_3 i3 (tri[0], tri[1], tri[2]);
 	i3.Sort();
 	boundaryfaces.PrepareSet (i3);
       }
@@ -1228,14 +1257,14 @@ namespace netgen
     for (i = 1; i <= mesh.GetNOpenElements(); i++)
       {
 	const Element2d & tri = mesh.OpenElement(i);
-	INDEX_3 i3 (tri.PNum(1), tri.PNum(2), tri.PNum(3));
+	INDEX_3 i3 (tri[0], tri[1], tri[2]);
 	i3.Sort();
 	boundaryfaces.Set (i3, 1);
       }
 
-    for (i = 1; i <= tempels.Size(); i++)
-      for (j = 1; j <= 4; j++)
-	tempels.Elem(i).NB(j) = 0;
+    for (i = 0; i < tempels.Size(); i++)
+      for (j = 0; j < 4; j++)
+	tempels[i].NB(j) = 0;
   
     TABLE<int,PointIndex::BASE> elsonpoint(mesh.GetNP());
     for (i = 0; i < tempels.Size(); i++)
@@ -1263,7 +1292,7 @@ namespace netgen
 
     INDEX_3_CLOSED_HASHTABLE<INDEX_2> faceht(100);   
   
-    Element2d hel(3);
+    Element2d hel(TRIG);
     for (pi = PointIndex::BASE; 
 	 pi < mesh.GetNP()+PointIndex::BASE; pi++)
       {
@@ -1275,13 +1304,13 @@ namespace netgen
 
 	    for (j = 1; j <= 4; j++)
 	      {
-		el.GetFace (j, hel);
+		el.GetFace1 (j, hel);
 		hel.Invert();
 		hel.NormalizeNumbering();
 	      
-		if (hel.PNum(1) == pi)
+		if (hel[0] == pi)
 		  {
-		    INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
+		    INDEX_3 i3(hel[0], hel[1], hel[2]);
 		  
 		    if (!boundaryfaces.Used (i3))
 		      {
@@ -1289,14 +1318,14 @@ namespace netgen
 			  {
 			    INDEX_2 i2 = faceht.Get(i3);
 			  
-			    tempels.Elem(i).NB(j) = i2.I1();
-			    tempels.Elem(i2.I1()).NB(i2.I2()) = i;
+			    tempels.Elem(i).NB1(j) = i2.I1();
+			    tempels.Elem(i2.I1()).NB1(i2.I2()) = i;
 			  }
 			else
 			  {
 			    hel.Invert();
 			    hel.NormalizeNumbering();
-			    INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
+			    INDEX_3 i3(hel[0], hel[1], hel[2]);
 			    INDEX_2 i2(i, j);
 			    faceht.Set (i3, i2);
 			  }
@@ -1314,7 +1343,7 @@ namespace netgen
       {
       INDEX_3 i3;
       Element2d face;
-      el.GetFace (j, face);
+      el.GetFace1 (j, face);
       for (int kk = 1; kk <= 3; kk++)
       i3.I(kk) = face.PNum(kk);
 
@@ -1346,7 +1375,7 @@ namespace netgen
       {
       (*testout) << i << " ";
       for (j = 1; j <= 4; j++)
-      (*testout) << tempels.Get(i).NB(j) << " ";
+      (*testout) << tempels.Get(i).NB1(j) << " ";
       (*testout) << endl;
       }
   
@@ -1415,10 +1444,10 @@ namespace netgen
 	if (done) break;
       
 	const DelaunayTet & el = tempels.Get(i);
-	const Point3d & p1 = mesh.Point (el.PNum(1));
-	const Point3d & p2 = mesh.Point (el.PNum(2));
-	const Point3d & p3 = mesh.Point (el.PNum(3));
-	const Point3d & p4 = mesh.Point (el.PNum(4));
+	const Point3d & p1 = mesh.Point (el[0]);
+	const Point3d & p2 = mesh.Point (el[1]);
+	const Point3d & p3 = mesh.Point (el[2]);
+	const Point3d & p4 = mesh.Point (el[3]);
       
 	Point3d ci = Center (p1, p2, p3, p4);
 
@@ -1449,16 +1478,18 @@ namespace netgen
 
 		for (j = 1; j <= 4; j++)
 		  {
-		    INDEX_3 i3;
+		    INDEX_3 i3 = tempels.Get(ei).GetFace1(j);
+		    /*
 		    Element2d face;
 		    tempels.Get(ei).GetFace(j, face);
 		    for (int kk = 1; kk <= 3; kk++)
 		      i3.I(kk) = face.PNum(kk);
+		    */
 		    i3.Sort();
 		  
 
-		    if (tempels.Get(ei).NB(j))
-		      elstack.Append (tempels.Get(ei).NB(j));
+		    if (tempels.Get(ei).NB1(j))
+		      elstack.Append (tempels.Get(ei).NB1(j));
 
 		    /*
 		      if (innerfaces.Used(i3))
@@ -1466,7 +1497,7 @@ namespace netgen
 		      INDEX_2 i2 = innerfaces.Get(i3);
 		      int other = i2.I1() + i2.I2() - ei;
 
-		      if (other != tempels.Get(ei).NB(j))
+		      if (other != tempels.Get(ei).NB1(j))
 		      cerr << "different1 !!" << endl;
 
 		      if (other)
@@ -1475,7 +1506,7 @@ namespace netgen
 		      }
 		      }
 		      else
-		      if (tempels.Get(ei).NB(j))
+		      if (tempels.Get(ei).NB1(j))
 		      cerr << "different2 !!" << endl;
 		    */
 
@@ -1492,10 +1523,10 @@ namespace netgen
 	for (i = 1; i <= ne; i++)
 	  {
 	    const DelaunayTet & el = tempels.Get(i);
-	    const Point3d & p1 = mesh.Point (el.PNum(1));
-	    const Point3d & p2 = mesh.Point (el.PNum(2));
-	    const Point3d & p3 = mesh.Point (el.PNum(3));
-	    const Point3d & p4 = mesh.Point (el.PNum(4));
+	    const Point3d & p1 = mesh.Point (el[0]);
+	    const Point3d & p2 = mesh.Point (el[1]);
+	    const Point3d & p3 = mesh.Point (el[2]);
+	    const Point3d & p4 = mesh.Point (el[3]);
 	  
 	    Point3d ci = Center (p1, p2, p3, p4);
 	  
@@ -1636,8 +1667,8 @@ namespace netgen
     for (i = 1; i <= tempels.Size(); i++)
       {
 	Element el(4);
-	for (j = 1; j <= 4; j++)
-	  el.PNum(j) = tempels.Elem(i).PNum(j);
+	for (j = 0; j < 4; j++)
+	  el[j] = tempels.Elem(i)[j];
 	mesh.AddVolumeElement (el);
       }
 
diff --git a/Netgen/libsrc/meshing/findip.hpp b/Netgen/libsrc/meshing/findip.hpp
index b47c3ac6dc..a0aed91b6c 100644
--- a/Netgen/libsrc/meshing/findip.hpp
+++ b/Netgen/libsrc/meshing/findip.hpp
@@ -1,13 +1,5 @@
-/*
 // find inner point
 
-#include <mystdlib.h>
-#include "meshing.hpp"
-
-namespace netgen
-{
-*/
-
 template <typename POINTARRAY, typename FACEARRAY>
 inline int FindInnerPoint (POINTARRAY & points,
 		    FACEARRAY & faces,
diff --git a/Netgen/libsrc/meshing/geomsearch.cpp b/Netgen/libsrc/meshing/geomsearch.cpp
index 53c04e12ac..1b4641308a 100644
--- a/Netgen/libsrc/meshing/geomsearch.cpp
+++ b/Netgen/libsrc/meshing/geomsearch.cpp
@@ -4,262 +4,260 @@
 
 namespace netgen
 {
-GeomSearch3d :: GeomSearch3d() 
-{
-  size.i1 = 0; size.i2 = 0; size.i3 = 0; 
-};
+  GeomSearch3d :: GeomSearch3d() 
+  {
+    size.i1 = 0; size.i2 = 0; size.i3 = 0; 
+  };
 
-GeomSearch3d :: ~GeomSearch3d()
-{
-  //delete old Hashtable:
-  if (size.i1 != 0)
-    {
-      for (int i = 1; i <= size.i1*size.i2*size.i3; i++)
-	{
-	  delete hashtable.Get(i);
-	}
-    } 
-}
+  GeomSearch3d :: ~GeomSearch3d()
+  {
+    //delete old Hashtable:
+    if (size.i1 != 0)
+      {
+	for (int i = 0; i < size.i1*size.i2*size.i3; i++)
+	  delete hashtable[i];
+      } 
+  }
 
-void GeomSearch3d :: Init (ARRAY <FrontPoint3,PointIndex::BASE> *pointsi, ARRAY <FrontFace> *facesi)
-{
-  points = pointsi;
-  faces = facesi;
-  size.i1 = 0; size.i2 = 0; size.i3 = 0; 
-  reset = 1;
-  hashcount = 1;
-}
+  void GeomSearch3d :: Init (ARRAY <FrontPoint3,PointIndex::BASE> *pointsi, ARRAY <FrontFace> *facesi)
+  {
+    points = pointsi;
+    faces = facesi;
+    size.i1 = 0; size.i2 = 0; size.i3 = 0; 
+    reset = 1;
+    hashcount = 1;
+  }
 
-void GeomSearch3d :: ElemMaxExt(Point3d& minp, Point3d& maxp, const Element2d& elem)
-{
-  maxp.X()=points->Get(elem.PNum(1)).P().X();
-  maxp.Y()=points->Get(elem.PNum(1)).P().Y();
-  maxp.Z()=points->Get(elem.PNum(1)).P().Z();
-  minp.X()=points->Get(elem.PNum(1)).P().X();
-  minp.Y()=points->Get(elem.PNum(1)).P().Y();
-  minp.Z()=points->Get(elem.PNum(1)).P().Z();
+  void GeomSearch3d :: ElemMaxExt(Point3d& minp, Point3d& maxp, const Element2d& elem)
+  {
+    maxp.X()=(*points)[elem.PNum(1)].P().X();
+    maxp.Y()=(*points)[elem.PNum(1)].P().Y();
+    maxp.Z()=(*points)[elem.PNum(1)].P().Z();
+    minp.X()=(*points)[elem.PNum(1)].P().X();
+    minp.Y()=(*points)[elem.PNum(1)].P().Y();
+    minp.Z()=(*points)[elem.PNum(1)].P().Z();
   
-  for (int i=2; i <= 3; i++)
-    {
-      maxp.X()=max2(points->Get(elem.PNum(i)).P().X(),maxp.X());
-      maxp.Y()=max2(points->Get(elem.PNum(i)).P().Y(),maxp.Y());
-      maxp.Z()=max2(points->Get(elem.PNum(i)).P().Z(),maxp.Z());
-      minp.X()=min2(points->Get(elem.PNum(i)).P().X(),minp.X());
-      minp.Y()=min2(points->Get(elem.PNum(i)).P().Y(),minp.Y());
-      minp.Z()=min2(points->Get(elem.PNum(i)).P().Z(),minp.Z());
-    }
-}
+    for (int i=2; i <= 3; i++)
+      {
+	maxp.X()=max2((*points)[elem.PNum(i)].P().X(),maxp.X());
+	maxp.Y()=max2((*points)[elem.PNum(i)].P().Y(),maxp.Y());
+	maxp.Z()=max2((*points)[elem.PNum(i)].P().Z(),maxp.Z());
+	minp.X()=min2((*points)[elem.PNum(i)].P().X(),minp.X());
+	minp.Y()=min2((*points)[elem.PNum(i)].P().Y(),minp.Y());
+	minp.Z()=min2((*points)[elem.PNum(i)].P().Z(),minp.Z());
+      }
+  }
 
-void GeomSearch3d :: MinCoords(const Point3d& p1, Point3d& p2)
-{
-  p2.X()=min2(p1.X(),p2.X());
-  p2.Y()=min2(p1.Y(),p2.Y());
-  p2.Z()=min2(p1.Z(),p2.Z());
-}
+  void GeomSearch3d :: MinCoords(const Point3d& p1, Point3d& p2)
+  {
+    p2.X()=min2(p1.X(),p2.X());
+    p2.Y()=min2(p1.Y(),p2.Y());
+    p2.Z()=min2(p1.Z(),p2.Z());
+  }
 
-void GeomSearch3d :: MaxCoords(const Point3d& p1, Point3d& p2)
-{
-  p2.X()=max2(p1.X(),p2.X());
-  p2.Y()=max2(p1.Y(),p2.Y());
-  p2.Z()=max2(p1.Z(),p2.Z());
-}
+  void GeomSearch3d :: MaxCoords(const Point3d& p1, Point3d& p2)
+  {
+    p2.X()=max2(p1.X(),p2.X());
+    p2.Y()=max2(p1.Y(),p2.Y());
+    p2.Z()=max2(p1.Z(),p2.Z());
+  }
 
-void GeomSearch3d :: Create()
-{
-  INDEX i,j,k;
-  if (reset)
-    {
-      const double hashelemsizefactor = 4;
-      reset = 0;
-      /*
-      minext=Point3d(MAXDOUBLE, MAXDOUBLE, MAXDOUBLE);
-      maxext=Point3d(MINDOUBLE, MINDOUBLE, MINDOUBLE);
-      */
-      ElemMaxExt(minext, maxext, faces->Get(1).Face());
-      Point3d maxp, minp;
-      Vec3d midext(0,0,0);
+  void GeomSearch3d :: Create()
+  {
+    INDEX i,j,k;
+    if (reset)
+      {
+	const double hashelemsizefactor = 4;
+	reset = 0;
+	/*
+	  minext=Point3d(MAXDOUBLE, MAXDOUBLE, MAXDOUBLE);
+	  maxext=Point3d(MINDOUBLE, MINDOUBLE, MINDOUBLE);
+	*/
+	ElemMaxExt(minext, maxext, faces->Get(1).Face());
+	Point3d maxp, minp;
+	Vec3d midext(0,0,0);
       
-      //get max Extension of Frontfaces
-      for (i = 1; i <= faces->Size(); i++)
-	{
-	  ElemMaxExt(minp, maxp, faces->Get(i).Face());
-	  MinCoords(minp, minext);
-	  MaxCoords(maxp, maxext);
-	  midext+=maxp-minp;
-	}
+	//get max Extension of Frontfaces
+	for (i = 1; i <= faces->Size(); i++)
+	  {
+	    ElemMaxExt(minp, maxp, faces->Get(i).Face());
+	    MinCoords(minp, minext);
+	    MaxCoords(maxp, maxext);
+	    midext+=maxp-minp;
+	  }
 
 
-      maxextreal = maxext;
-      maxext = maxext + 1e-4 * (maxext - minext);
+	maxextreal = maxext;
+	maxext = maxext + 1e-4 * (maxext - minext);
 
-      midext*=1./faces->Size();
-      Vec3d boxext = maxext - minext;
+	midext*=1./faces->Size();
+	Vec3d boxext = maxext - minext;
       
-      //delete old Hashtable:
-      if (size.i1 != 0)
-	{
-	  for (i = 1; i <= size.i1*size.i2*size.i3; i++)
-	    {
-	      delete hashtable.Get(i);
-	    }
-	} 
+	//delete old Hashtable:
+	if (size.i1 != 0)
+	  {
+	    for (i = 1; i <= size.i1*size.i2*size.i3; i++)
+	      {
+		delete hashtable.Get(i);
+	      }
+	  } 
       
-      size.i1 = int (boxext.X()/midext.X()/hashelemsizefactor+1);
-      size.i2 = int (boxext.Y()/midext.Y()/hashelemsizefactor+1);
-      size.i3 = int (boxext.Z()/midext.Z()/hashelemsizefactor+1);
-      PrintMessage (5, "hashsizes = ", size.i1, ", ", size.i2, ", ", size.i3);
+	size.i1 = int (boxext.X()/midext.X()/hashelemsizefactor+1);
+	size.i2 = int (boxext.Y()/midext.Y()/hashelemsizefactor+1);
+	size.i3 = int (boxext.Z()/midext.Z()/hashelemsizefactor+1);
+	PrintMessage (5, "hashsizes = ", size.i1, ", ", size.i2, ", ", size.i3);
       
-      elemsize.X()=boxext.X()/size.i1;
-      elemsize.Y()=boxext.Y()/size.i2;
-      elemsize.Z()=boxext.Z()/size.i3;
+	elemsize.X()=boxext.X()/size.i1;
+	elemsize.Y()=boxext.Y()/size.i2;
+	elemsize.Z()=boxext.Z()/size.i3;
 
-      //create Hasharrays:
-      hashtable.SetSize(size.i1*size.i2*size.i3);
-      for (i = 1; i <= size.i1; i++)
-	{
-	  for (j = 1; j <= size.i2; j++)
-	    {
-	      for (k = 1; k <= size.i3; k++)
-		{
-		  INDEX ind=i+(j-1)*size.i1+(k-1)*size.i2*size.i1;
-		  hashtable.Elem(ind) = new ARRAY <int> ();
-		}
-	    }
-	}
-    }
-  else
-    {
-      //Clear all Hash-Arrays
-      for (i = 1; i <= size.i1; i++)
-	{
-	  for (j = 1; j <= size.i2; j++)
-	    {
-	      for (k = 1; k <= size.i3; k++)
-		{
-		  INDEX ind=i+(j-1)*size.i1+(k-1)*size.i2*size.i1;
-		  hashtable.Elem(ind)->SetSize(0);
-		}
-	    }
-	}	  
-    }
+	//create Hasharrays:
+	hashtable.SetSize(size.i1*size.i2*size.i3);
+	for (i = 1; i <= size.i1; i++)
+	  {
+	    for (j = 1; j <= size.i2; j++)
+	      {
+		for (k = 1; k <= size.i3; k++)
+		  {
+		    INDEX ind=i+(j-1)*size.i1+(k-1)*size.i2*size.i1;
+		    hashtable.Elem(ind) = new ARRAY <int> ();
+		  }
+	      }
+	  }
+      }
+    else
+      {
+	//Clear all Hash-Arrays
+	for (i = 1; i <= size.i1; i++)
+	  {
+	    for (j = 1; j <= size.i2; j++)
+	      {
+		for (k = 1; k <= size.i3; k++)
+		  {
+		    INDEX ind=i+(j-1)*size.i1+(k-1)*size.i2*size.i1;
+		    hashtable.Elem(ind)->SetSize(0);
+		  }
+	      }
+	  }	  
+      }
   
-  //Faces in Hashtable einfuegen:
-  for (i = 1; i <= faces->Size(); i++)
-    {
-      AddElem(faces->Get(i).Face(),i);
-    }
+    //Faces in Hashtable einfuegen:
+    for (i = 1; i <= faces->Size(); i++)
+      {
+	AddElem(faces->Get(i).Face(),i);
+      }
   
-}
+  }
 
-void GeomSearch3d :: AddElem(const Element2d& elem, INDEX elemnum)
-{
-  Point3d minp, maxp;
-  ElemMaxExt(minp, maxp, elem);
-  int sx = int ((minp.X()-minext.X())/elemsize.X()+1.);
-  int ex = int ((maxp.X()-minext.X())/elemsize.X()+1.);
-  int sy = int ((minp.Y()-minext.Y())/elemsize.Y()+1.);
-  int ey = int ((maxp.Y()-minext.Y())/elemsize.Y()+1.);
-  int sz = int ((minp.Z()-minext.Z())/elemsize.Z()+1.);
-  int ez = int ((maxp.Z()-minext.Z())/elemsize.Z()+1.);
-  int ix,iy,iz;
+  void GeomSearch3d :: AddElem(const Element2d& elem, INDEX elemnum)
+  {
+    Point3d minp, maxp;
+    ElemMaxExt(minp, maxp, elem);
+    int sx = int ((minp.X()-minext.X())/elemsize.X()+1.);
+    int ex = int ((maxp.X()-minext.X())/elemsize.X()+1.);
+    int sy = int ((minp.Y()-minext.Y())/elemsize.Y()+1.);
+    int ey = int ((maxp.Y()-minext.Y())/elemsize.Y()+1.);
+    int sz = int ((minp.Z()-minext.Z())/elemsize.Z()+1.);
+    int ez = int ((maxp.Z()-minext.Z())/elemsize.Z()+1.);
+    int ix,iy,iz;
   
-  for (ix = sx; ix <= ex; ix++)
-    {
-      for (iy = sy; iy <= ey; iy++)
-	{
-	  for (iz = sz; iz <= ez; iz++)
-	    {
-	      INDEX ind=ix+(iy-1)*size.i1+(iz-1)*size.i2*size.i1;
-	      if (ind < 1 || ind > size.i1 * size.i2 * size.i3)
-		{
-		  cerr << "Illegal hash-position";
-		  cerr << "Position: " << ix << "," << iy << "," << iz << endl;
-		}
-	      hashtable.Elem(ind)->Append(elemnum);		      
-	    }
-	}
-    }
-}
+    for (ix = sx; ix <= ex; ix++)
+      {
+	for (iy = sy; iy <= ey; iy++)
+	  {
+	    for (iz = sz; iz <= ez; iz++)
+	      {
+		INDEX ind=ix+(iy-1)*size.i1+(iz-1)*size.i2*size.i1;
+		if (ind < 1 || ind > size.i1 * size.i2 * size.i3)
+		  {
+		    cerr << "Illegal hash-position";
+		    cerr << "Position: " << ix << "," << iy << "," << iz << endl;
+		  }
+		hashtable.Elem(ind)->Append(elemnum);		      
+	      }
+	  }
+      }
+  }
 
-void GeomSearch3d :: GetLocals(ARRAY<Element2d> & locfaces,  ARRAY<INDEX> & findex,
-			       INDEX fstind, const Point3d& p0, double xh)
-{
-  hashcount++;
+  void GeomSearch3d :: GetLocals(ARRAY<Element2d> & locfaces,  ARRAY<INDEX> & findex,
+				 INDEX fstind, const Point3d& p0, double xh)
+  {
+    hashcount++;
   
-  Point3d minp, maxp, midp; 
+    Point3d minp, maxp, midp; 
 
-  minp=p0-Vec3d(xh,xh,xh); //lay cube over sphere
-  maxp=p0+Vec3d(xh,xh,xh);
+    minp=p0-Vec3d(xh,xh,xh); //lay cube over sphere
+    maxp=p0+Vec3d(xh,xh,xh);
 
-  MaxCoords(minext,minp); //cube may not be out of hash-region
-  MinCoords(maxextreal,maxp);
+    MaxCoords(minext,minp); //cube may not be out of hash-region
+    MinCoords(maxextreal,maxp);
 
 
-  int cluster = faces->Get(fstind).Cluster();
+    int cluster = faces->Get(fstind).Cluster();
   
-  int sx = int((minp.X()-minext.X())/elemsize.X()+1.);
-  int ex = int((maxp.X()-minext.X())/elemsize.X()+1.);
-  int sy = int((minp.Y()-minext.Y())/elemsize.Y()+1.);
-  int ey = int((maxp.Y()-minext.Y())/elemsize.Y()+1.);
-  int sz = int((minp.Z()-minext.Z())/elemsize.Z()+1.);
-  int ez = int((maxp.Z()-minext.Z())/elemsize.Z()+1.);
-  int ix,iy,iz,i,k;
+    int sx = int((minp.X()-minext.X())/elemsize.X()+1.);
+    int ex = int((maxp.X()-minext.X())/elemsize.X()+1.);
+    int sy = int((minp.Y()-minext.Y())/elemsize.Y()+1.);
+    int ey = int((maxp.Y()-minext.Y())/elemsize.Y()+1.);
+    int sz = int((minp.Z()-minext.Z())/elemsize.Z()+1.);
+    int ez = int((maxp.Z()-minext.Z())/elemsize.Z()+1.);
+    int ix,iy,iz,i,k;
 
-  int cnt1 = 0;  // test, how efficient hashtable is
-  int cnt2 = 0;
-  int cnt3 = 0;
+    int cnt1 = 0;  // test, how efficient hashtable is
+    int cnt2 = 0;
+    int cnt3 = 0;
   
-  for (ix = sx; ix <= ex; ix++)
-    {
-      for (iy = sy; iy <= ey; iy++)
-	{
-	  for (iz = sz; iz <= ez; iz++)
-	    {
-	      INDEX ind=ix+(iy-1)*size.i1+(iz-1)*size.i2*size.i1;
+    for (ix = sx; ix <= ex; ix++)
+      {
+	for (iy = sy; iy <= ey; iy++)
+	  {
+	    for (iz = sz; iz <= ez; iz++)
+	      {
+		INDEX ind=ix+(iy-1)*size.i1+(iz-1)*size.i2*size.i1;
 	      
-	      //go through all elements in one hash area
-	      const ARRAY <int> & area = *hashtable.Elem(ind);
-	      for (k = 1; k <= area.Size(); k++)
-		{
-		  cnt2++;
-		  i = area.Get(k);
-		  if (faces->Get(i).Cluster() == cluster && 
-		      faces->Get(i).Valid() &&
-		      faces->Get(i).HashValue() != hashcount && 
-		      i != fstind)
-		    {
-		      cnt1++;
-		      const Element2d & face = faces->Get(i).Face();
+		//go through all elements in one hash area
+		const ARRAY <int> & area = *hashtable.Elem(ind);
+		for (k = 1; k <= area.Size(); k++)
+		  {
+		    cnt2++;
+		    i = area.Get(k);
+		    if (faces->Get(i).Cluster() == cluster && 
+			faces->Get(i).Valid() &&
+			faces->Get(i).HashValue() != hashcount && 
+			i != fstind)
+		      {
+			cnt1++;
+			const Element2d & face = faces->Get(i).Face();
 		      
-		      const Point3d & p1 = points->Get(face.PNum(1)).P();
-		      const Point3d & p2 = points->Get(face.PNum(2)).P();
-		      const Point3d & p3 = points->Get(face.PNum(3)).P();
+			const Point3d & p1 = (*points)[face.PNum(1)].P();
+			const Point3d & p2 = (*points)[face.PNum(2)].P();
+			const Point3d & p3 = (*points)[face.PNum(3)].P();
 		      
-		      midp = Center (p1, p2, p3);
+			midp = Center (p1, p2, p3);
 		      
-		      if (Dist2 (midp, p0) <= xh*xh)
-			{
-			  cnt3++;
-			  locfaces.Append(faces->Get(i).Face());
-			  findex.Append(i);
-			  faces->Elem(i).SetHashValue(hashcount);
-			}
-		    }
-		}
-	    }
-	}
-    }
-  /*
-  if (faces->Size() != 0 && hashcount % 200 == 0)
-    {
+			if (Dist2 (midp, p0) <= xh*xh)
+			  {
+			    cnt3++;
+			    locfaces.Append(faces->Get(i).Face());
+			    findex.Append(i);
+			    faces->Elem(i).SetHashValue(hashcount);
+			  }
+		      }
+		  }
+	      }
+	  }
+      }
+    /*
+      if (faces->Size() != 0 && hashcount % 200 == 0)
+      {
       (*mycout) << "n.o.f= " << faces->Size();
       (*mycout) << ", n.o.lf= " << locfaces.Size();
       (*mycout) << ", hashf= " << (double)cnt2/(double)faces->Size();
       (*mycout) << " (" << (double)cnt1/(double)faces->Size();
       (*mycout) << ", " << (double)cnt3/(double)faces->Size() << ")" << endl;
-    }
+      }
     */
 
-}
+  }
 
 }
diff --git a/Netgen/libsrc/meshing/global.cpp b/Netgen/libsrc/meshing/global.cpp
index b13367b1dc..60a1fbe51d 100644
--- a/Netgen/libsrc/meshing/global.cpp
+++ b/Netgen/libsrc/meshing/global.cpp
@@ -3,10 +3,10 @@
 
 namespace netgen
 {
-  ostream * testout;
+  ostream * testout = &cout;
 
-  ostream * mycout;
-  ostream * myerr;
+  ostream * mycout = &cout;
+  ostream * myerr = &cerr;
 
 
   // Flags globflags; // not used anymoure
diff --git a/Netgen/libsrc/meshing/hpref_prism.hpp b/Netgen/libsrc/meshing/hpref_prism.hpp
new file mode 100644
index 0000000000..ffbf2d2873
--- /dev/null
+++ b/Netgen/libsrc/meshing/hpref_prism.hpp
@@ -0,0 +1,300 @@
+
+
+
+
+
+  // HP_PRISM  ... no refinement
+  int refprism_splitedges[][3] =
+    {
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refprism_newelstypes[] =
+    {
+      HP_PRISM,
+      HP_NONE,
+    };
+  int refprism_newels[][8] =
+    {
+      { 1, 2, 3, 4, 5, 6 }
+    };
+  HPRef_Struct refprism =
+    {
+      HP_PRISM,
+      refprism_splitedges, 
+      0, 0,
+      refprism_newelstypes, 
+      refprism_newels
+    };
+
+
+
+  // HP_PRISM_SINGEDGE  ... vertical edge 1-4 is singular
+  int refprism_singedge_splitedges[][3] =
+    {
+      { 1, 2, 7 },
+      { 1, 3, 8 },
+      { 4, 5, 9 },
+      { 4, 6, 10 },
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refprism_singedge_newelstypes[] =
+    {
+      HP_PRISM_SINGEDGE,
+      HP_HEX,
+      HP_NONE,
+    };
+  int refprism_singedge_newels[][8] =
+    {
+      { 1, 7, 8, 4, 9, 10 },
+      { 3, 8, 7, 2, 6, 10, 9, 5 }
+    };
+  HPRef_Struct refprism_singedge =
+    {
+      HP_PRISM,
+      refprism_singedge_splitedges, 
+      0, 0,
+      refprism_singedge_newelstypes, 
+      refprism_singedge_newels
+    };
+
+
+
+
+
+
+  // HP_PRISM_SINGEDGE_V12  vertical edges 1-4 and 2-5 are singular 
+  int refprism_singedge_v12_splitedges[][3] =
+    {
+      { 1, 2, 7 },
+      { 1, 3, 8 },
+      { 2, 1, 9 },
+      { 2, 3, 10 },
+      { 4, 5, 11 },
+      { 4, 6, 12 },
+      { 5, 4, 13 },
+      { 5, 6, 14},
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refprism_singedge_v12_newelstypes[] =
+    {
+      HP_HEX,
+      HP_PRISM,
+      HP_PRISM,
+      HP_PRISM,
+      HP_NONE,
+    };
+  int refprism_singedge_v12_newels[][8] =
+    {
+      { 7, 9, 10, 8, 11, 13, 14, 12 },
+      { 1, 7, 8, 4, 11, 12 },
+      { 2, 10, 9, 5, 14, 13 },
+      { 3, 8, 10, 6, 12, 14 },
+    };
+  HPRef_Struct refprism_singedge_v12 =
+    {
+      HP_PRISM,
+      refprism_singedge_v12_splitedges, 
+      0, 0,
+      refprism_singedge_v12_newelstypes, 
+      refprism_singedge_v12_newels
+    };
+
+
+
+
+
+
+  // HP_PRISM_SINGEDGE_H12
+  int refprism_singedge_h12_splitedges[][3] =
+    {
+      { 1, 3, 7 },
+      { 2, 1, 8 },
+      { 2, 3, 9 },
+      { 3, 1, 10 },
+
+      { 4, 6, 12 },
+      { 5, 4, 13 },
+      { 5, 6, 14 },
+      { 6, 4, 15 },
+
+      { 0, 0, 0 }
+    };
+
+  int refprism_singedge_h12_splitfaces[][4] =
+    {
+      { 2, 1, 3, 11 },
+      { 5, 4, 6, 16 },
+      { 0, 0, 0, 0 },
+    };
+
+  HPREF_ELEMENT_TYPE refprism_singedge_h12_newelstypes[] =
+    {
+      HP_HEX,
+      HP_HEX,
+      HP_PRISM,
+      HP_PRISM,
+      HP_PRISM,
+      HP_NONE,
+    };
+  int refprism_singedge_h12_newels[][8] =
+    {
+      { 1, 8, 11, 7, 4, 13, 16, 12 },
+      { 9, 3, 10, 11, 14, 6, 15, 16 },
+      { 7, 11, 10, 12, 16, 15 },
+      { 2, 9, 11, 5, 14, 16 },
+      { 8, 2, 11, 13, 5, 16 }
+    };
+  HPRef_Struct refprism_singedge_h12 =
+    {
+      HP_PRISM,
+      refprism_singedge_h12_splitedges, 
+      refprism_singedge_h12_splitfaces, 
+      0,
+      refprism_singedge_h12_newelstypes, 
+      refprism_singedge_h12_newels
+    };
+
+
+
+
+
+
+  // HP_PRISM_SINGEDGE_H1
+  int refprism_singedge_h1_splitedges[][3] =
+    {
+      { 1, 3, 7 },
+      { 2, 3, 8 },
+      { 4, 6, 9 },
+      { 5, 6, 10 },
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refprism_singedge_h1_newelstypes[] =
+    {
+      HP_HEX,
+      HP_PRISM,
+      HP_NONE,
+    };
+  int refprism_singedge_h1_newels[][8] =
+    {
+      { 1, 2, 8, 7, 4, 5, 10, 9 },
+      { 3, 7, 8, 6, 9, 10 }
+    };
+  HPRef_Struct refprism_singedge_h1 =
+    {
+      HP_PRISM,
+      refprism_singedge_h1_splitedges, 
+      0, 0,
+      refprism_singedge_h1_newelstypes, 
+      refprism_singedge_h1_newels
+    };
+
+
+
+
+
+
+
+//  HP_PRISM_1FA_0E_0V
+  int refprism_1fa_0e_0v_splitedges[][3] =
+    {
+      { 1, 4, 7 },
+      { 2, 5, 8 },
+      { 3, 6, 9 },
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refprism_1fa_0e_0v_newelstypes[] =
+    {
+      HP_PRISM_1FA_0E_0V,
+      HP_PRISM,
+      HP_NONE,
+    };
+  int refprism_1fa_0e_0v_newels[][8] =
+    {
+      { 1, 2, 3, 7, 8, 9 },
+      { 7, 8, 9, 4, 5, 6 }
+    };
+  HPRef_Struct refprism_1fa_0e_0v =
+    {
+      HP_PRISM,
+      refprism_1fa_0e_0v_splitedges, 
+      0, 0,
+      refprism_1fa_0e_0v_newelstypes, 
+      refprism_1fa_0e_0v_newels
+    };
+
+
+
+
+
+
+//  HP_PRISM_1FB_0E_0V   ... quad face 1-2-4-5
+  int refprism_1fb_0e_0v_splitedges[][3] =
+    {
+      { 1, 3, 7 },
+      { 2, 3, 8 },
+      { 4, 6, 9 },
+      { 5, 6, 10 },
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refprism_1fb_0e_0v_newelstypes[] =
+    {
+      HP_HEX_1F_0E_0V,
+      HP_PRISM,
+      HP_NONE,
+    };
+  int refprism_1fb_0e_0v_newels[][8] =
+    {
+      { 1, 4, 5, 2, 7, 9, 10, 8  },
+      { 7, 8, 3, 9, 10, 6 }
+    };
+  HPRef_Struct refprism_1fb_0e_0v =
+    {
+      HP_PRISM,
+      refprism_1fb_0e_0v_splitedges, 
+      0, 0,
+      refprism_1fb_0e_0v_newelstypes, 
+      refprism_1fb_0e_0v_newels
+    };
+
+
+
+
+
+
+//  HP_PRISM_1FB_1EA_0V   ... quad face 1-2-4-5
+  int refprism_1fb_1ea_0v_splitedges[][3] =
+    {
+      { 1, 3, 7 },
+      { 2, 3, 8 },
+      { 4, 6, 9 },
+      { 5, 6, 10 },
+      { 1, 2, 11 },
+      { 4, 5, 12 },
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refprism_1fb_1ea_0v_newelstypes[] =
+    {
+      HP_HEX_1F_0E_0V,
+      HP_PRISM_1FB_1EA_0V,
+      HP_PRISM,
+      HP_NONE,
+    };
+  int refprism_1fb_1ea_0v_newels[][8] =
+    {
+      { 11, 12, 5, 2, 7, 9, 10, 8  },
+      { 1, 11, 7, 4, 12, 9 },
+      { 7, 8, 3, 9, 10, 6 }
+    };
+  HPRef_Struct refprism_1fb_1ea_0v =
+    {
+      HP_PRISM,
+      refprism_1fb_1ea_0v_splitedges, 
+      0, 0,
+      refprism_1fb_1ea_0v_newelstypes, 
+      refprism_1fb_1ea_0v_newels
+    };
+
+
+
+
+
diff --git a/Netgen/libsrc/meshing/hpref_quad.hpp b/Netgen/libsrc/meshing/hpref_quad.hpp
new file mode 100644
index 0000000000..8802d4539e
--- /dev/null
+++ b/Netgen/libsrc/meshing/hpref_quad.hpp
@@ -0,0 +1,2129 @@
+
+
+
+
+// HP_QUAD
+int refquad_splitedges[][3] =
+{
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_newelstypes[] =
+{
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_newels[][8] =
+{
+  { 1, 2, 3, 4 },
+};
+HPRef_Struct refquad =
+{
+  HP_QUAD,
+  refquad_splitedges, 
+  0, 0,
+  refquad_newelstypes, 
+  refquad_newels
+};
+
+
+
+
+
+
+
+// HP_QUAD_SINGCORNER
+int refquad_singcorner_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_singcorner_newelstypes[] =
+{
+  HP_TRIG_SINGCORNER,
+  HP_QUAD,
+  HP_TRIG,
+  HP_NONE,
+};
+int refquad_singcorner_newels[][8] =
+{
+  { 1, 5, 6 },
+  { 2, 4, 6, 5 },
+  { 2, 3, 4 },
+};
+HPRef_Struct refquad_singcorner =
+{
+  HP_QUAD,
+  refquad_singcorner_splitedges, 
+  0, 0,
+  refquad_singcorner_newelstypes, 
+  refquad_singcorner_newels
+};
+
+
+
+
+
+// HP_DUMMY_QUAD_SINGCORNER
+int refdummyquad_singcorner_splitedges[][3] =
+{
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refdummyquad_singcorner_newelstypes[] =
+{
+  HP_TRIG_SINGCORNER,
+  HP_TRIG,
+  HP_NONE,
+};
+int refdummyquad_singcorner_newels[][8] =
+{
+  { 1, 2, 4 },
+  { 4, 2, 3 },
+};
+HPRef_Struct refdummyquad_singcorner =
+{
+  HP_QUAD,
+  refdummyquad_singcorner_splitedges, 
+  0, 0,
+  refdummyquad_singcorner_newelstypes, 
+  refdummyquad_singcorner_newels
+};
+
+
+
+
+
+
+
+// HP_QUAD_SINGEDGE
+int refquad_singedge_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_singedge_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_singedge_newels[][8] =
+{
+  { 1, 2, 6, 5 },
+  { 5, 6, 3, 4 },
+};
+HPRef_Struct refquad_singedge =
+{
+  HP_QUAD,
+  refquad_singedge_splitedges, 
+  0, 0,
+  refquad_singedge_newelstypes, 
+  refquad_singedge_newels
+};
+
+
+
+
+
+
+// HP_QUAD_0E_2VA
+int refquad_0e_2va_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 1, 7 },
+  { 2, 3, 8 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_0e_2va_newelstypes[] =
+{
+  HP_TRIG_SINGCORNER,
+  HP_TRIG_SINGCORNER,
+  HP_QUAD,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_0e_2va_newels[][8] =
+{
+  { 1, 5, 6 },
+  { 2, 8, 7 },
+  { 5, 7, 8, 6 },
+  { 6, 8, 3, 4 },
+};
+HPRef_Struct refquad_0e_2va =
+{
+  HP_QUAD,
+  refquad_0e_2va_splitedges, 
+  0, 0,
+  refquad_0e_2va_newelstypes, 
+  refquad_0e_2va_newels
+};
+
+
+
+// HP_QUAD_0E_2VB
+int refquad_0e_2vb_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 3, 4, 7 },
+  { 3, 2, 8 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_0e_2vb_newelstypes[] =
+{
+  HP_TRIG_SINGCORNER,
+  HP_TRIG_SINGCORNER,
+  HP_QUAD,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_0e_2vb_newels[][8] =
+{
+  { 1, 5, 6 },
+  { 3, 7, 8 },
+  { 5, 2, 4, 6 },
+  { 2, 8, 7, 4 },
+};
+HPRef_Struct refquad_0e_2vb =
+{
+  HP_QUAD,
+  refquad_0e_2vb_splitedges, 
+  0, 0,
+  refquad_0e_2vb_newelstypes, 
+  refquad_0e_2vb_newels
+};
+
+
+
+
+// HP_QUAD_0E_3V
+int refquad_0e_3v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 1, 7 },
+  { 2, 3, 8 },
+  { 3, 2, 9 },
+  { 3, 4, 10 },
+  { 0, 0, 0 }
+};
+
+int refquad_0e_3v_splitfaces[][4] =
+{
+  { 2, 3, 1, 14 },
+  { 0, 0, 0, 0 },
+};
+
+HPREF_ELEMENT_TYPE refquad_0e_3v_newelstypes[] =
+{
+  HP_TRIG_SINGCORNER,
+  HP_DUMMY_QUAD_SINGCORNER,
+  HP_TRIG_SINGCORNER,
+  HP_QUAD,
+  HP_QUAD,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_0e_3v_newels[][8] =
+{
+  { 1, 5, 6 },
+  { 2, 8, 14, 7 },
+  { 3, 10, 9 },
+  { 5, 7, 14, 6 },
+  { 8, 9, 10, 14 },
+  { 6, 14, 10, 4 },
+};
+HPRef_Struct refquad_0e_3v =
+{
+  HP_QUAD,
+  refquad_0e_3v_splitedges, 
+  refquad_0e_3v_splitfaces, 
+  0,
+  refquad_0e_3v_newelstypes, 
+  refquad_0e_3v_newels
+};
+
+
+
+
+// HP_QUAD_0E_4V
+int refquad_0e_4v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 1, 7 },
+  { 2, 3, 8 },
+  { 3, 2, 9 },
+  { 3, 4, 10 },
+  { 4, 1, 11 },
+  { 4, 3, 12 },
+  { 0, 0, 0 }
+};
+
+int refquad_0e_4v_splitfaces[][4] =
+{
+  { 1, 2, 4, 13 },
+  { 2, 3, 1, 14 },
+  { 3, 4, 2, 15 },
+  { 4, 1, 3, 16 },
+  { 0, 0, 0, 0 },
+};
+
+HPREF_ELEMENT_TYPE refquad_0e_4v_newelstypes[] =
+{
+  HP_DUMMY_QUAD_SINGCORNER,
+  HP_DUMMY_QUAD_SINGCORNER,
+  HP_DUMMY_QUAD_SINGCORNER,
+  HP_DUMMY_QUAD_SINGCORNER,
+
+  HP_QUAD,
+  HP_QUAD,
+  HP_QUAD,
+  HP_QUAD,
+
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_0e_4v_newels[][8] =
+{
+  { 1, 5, 13, 6 },
+  { 2, 8, 14, 7 },
+  { 3, 10, 15, 9 },
+  { 4, 11, 16, 12 },
+  { 5, 7, 14, 13 },
+  { 8, 9, 15, 14 },
+  { 10, 12, 16, 15 },
+  { 11, 6, 13, 16 },
+  { 13, 14, 15, 16 }
+};
+HPRef_Struct refquad_0e_4v =
+{
+  HP_QUAD,
+  refquad_0e_4v_splitedges, 
+  refquad_0e_4v_splitfaces, 
+  0,
+  refquad_0e_4v_newelstypes, 
+  refquad_0e_4v_newels
+};
+
+
+
+
+
+
+
+
+// HP_QUAD_1E_1VA
+int refquad_1e_1va_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 1, 2, 7 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_1va_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_NONE,
+};
+int refquad_1e_1va_newels[][8] =
+{
+  { 7, 2, 6, 5 },
+  { 5, 6, 3, 4 },
+  { 1, 7, 5 },
+};
+HPRef_Struct refquad_1e_1va =
+{
+  HP_QUAD,
+  refquad_1e_1va_splitedges, 
+  0, 0,
+  refquad_1e_1va_newelstypes, 
+  refquad_1e_1va_newels
+};
+
+
+
+
+// HP_QUAD_1E_1VB
+int refquad_1e_1vb_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 2, 1, 7 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_1vb_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_NONE,
+};
+int refquad_1e_1vb_newels[][8] =
+{
+  { 1, 7, 6, 5 },
+  { 5, 6, 3, 4 },
+  { 7, 2, 6 },
+};
+HPRef_Struct refquad_1e_1vb =
+{
+  HP_QUAD,
+  refquad_1e_1vb_splitedges, 
+  0, 0,
+  refquad_1e_1vb_newelstypes, 
+  refquad_1e_1vb_newels
+};
+
+
+
+// HP_QUAD_1E_1VC
+int refquad_1e_1vc_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 3, 2, 7 },
+  { 3, 4, 8 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_1vc_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_TRIG,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_NONE,
+};
+int refquad_1e_1vc_newels[][8] =
+{
+  { 1, 2, 6, 5 },
+  { 5, 6, 4 },
+  { 4, 6, 7, 8 },
+  { 3, 8, 7 }
+};
+HPRef_Struct refquad_1e_1vc =
+{
+  HP_QUAD,
+  refquad_1e_1vc_splitedges, 
+  0, 0,
+  refquad_1e_1vc_newelstypes, 
+  refquad_1e_1vc_newels
+};
+
+
+
+// HP_QUAD_1E_1VD
+int refquad_1e_1vd_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 4, 1, 7 },
+  { 4, 3, 8 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_1vd_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_TRIG,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_NONE,
+};
+int refquad_1e_1vd_newels[][8] =
+{
+  { 1, 2, 6, 5 },
+  { 5, 6, 3 },
+  { 5, 3, 8, 7 },
+  { 4, 7, 8 }
+};
+HPRef_Struct refquad_1e_1vd =
+{
+  HP_QUAD,
+  refquad_1e_1vd_splitedges, 
+  0, 0,
+  refquad_1e_1vd_newelstypes, 
+  refquad_1e_1vd_newels
+};
+
+
+
+
+
+
+
+// HP_QUAD_1E_2VA
+int refquad_1e_2va_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 1, 2, 7 },
+  { 2, 1, 8 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_2va_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_NONE,
+};
+int refquad_1e_2va_newels[][8] =
+{
+  { 7, 8, 6, 5 },
+  { 5, 6, 3, 4 },
+  { 1, 7, 5 },
+  { 8, 2, 6 }
+};
+HPRef_Struct refquad_1e_2va =
+{
+  HP_QUAD,
+  refquad_1e_2va_splitedges, 
+  0, 0,
+  refquad_1e_2va_newelstypes, 
+  refquad_1e_2va_newels
+};
+
+
+
+
+// HP_QUAD_1E_2VB
+int refquad_1e_2vb_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 1, 2, 7 },
+  { 3, 2, 8 },
+  { 3, 4, 9 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_2vb_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_NONE,
+};
+int refquad_1e_2vb_newels[][8] =
+{
+  { 7, 2, 6, 5 },
+  { 1, 7, 5 },
+  { 5, 6, 4 },
+  { 4, 6, 8, 9 },
+  { 3, 9, 8 }
+};
+HPRef_Struct refquad_1e_2vb =
+{
+  HP_QUAD,
+  refquad_1e_2vb_splitedges, 
+  0, 0,
+  refquad_1e_2vb_newelstypes, 
+  refquad_1e_2vb_newels
+};
+
+
+
+
+// HP_QUAD_1E_2VC
+int refquad_1e_2vc_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 1, 2, 7 },
+  { 4, 1, 8 },
+  { 4, 3, 9 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_2vc_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_NONE,
+};
+int refquad_1e_2vc_newels[][8] =
+{
+  { 7, 2, 6, 5 },
+  { 1, 7, 5 },
+  { 5, 6, 3 },
+  { 5, 3, 9, 8 },
+  { 4, 8, 9 }
+};
+HPRef_Struct refquad_1e_2vc =
+{
+  HP_QUAD,
+  refquad_1e_2vc_splitedges, 
+  0, 0,
+  refquad_1e_2vc_newelstypes, 
+  refquad_1e_2vc_newels
+};
+
+
+
+
+// HP_QUAD_1E_2VD
+int refquad_1e_2vd_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 2, 1, 7 },
+  { 3, 2, 8 },
+  { 3, 4, 9 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_2vd_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_NONE,
+};
+int refquad_1e_2vd_newels[][8] =
+{
+  { 1, 7, 6, 5 },
+  { 7, 2, 6 },
+  { 5, 6, 4 },
+  { 4, 6, 8, 9 },
+  { 3, 9, 8 }
+};
+HPRef_Struct refquad_1e_2vd =
+{
+  HP_QUAD,
+  refquad_1e_2vd_splitedges, 
+  0, 0,
+  refquad_1e_2vd_newelstypes, 
+  refquad_1e_2vd_newels
+};
+
+
+
+
+
+// HP_QUAD_1E_2VE
+int refquad_1e_2ve_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 2, 1, 7 },
+  { 4, 1, 8 },
+  { 4, 3, 9 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_2ve_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_NONE,
+};
+int refquad_1e_2ve_newels[][8] =
+{
+  { 1, 7, 6, 5 },
+  { 7, 2, 6 },
+  { 5, 6, 3 },
+  { 5, 3, 9, 8 },
+  { 4, 8, 9 }
+};
+HPRef_Struct refquad_1e_2ve =
+{
+  HP_QUAD,
+  refquad_1e_2ve_splitedges, 
+  0, 0,
+  refquad_1e_2ve_newelstypes, 
+  refquad_1e_2ve_newels
+};
+
+
+
+
+
+
+// HP_QUAD_1E_2VF
+int refquad_1e_2vf_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 4, 1, 7 },
+  { 4, 3, 8 },
+  { 3, 2, 9 },
+  { 3, 4, 10 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_2vf_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_TRIG_SINGCORNER,
+  HP_NONE,
+};
+int refquad_1e_2vf_newels[][8] =
+{
+  { 1, 2, 6, 5 },
+  { 5, 6, 9, 7 },
+  { 7, 9, 10, 8 },
+  { 4, 7, 8 },
+  { 3, 10, 9 },
+};
+HPRef_Struct refquad_1e_2vf =
+{
+  HP_QUAD,
+  refquad_1e_2vf_splitedges, 
+  0, 0,
+  refquad_1e_2vf_newelstypes, 
+  refquad_1e_2vf_newels
+};
+
+
+
+
+
+// HP_QUAD_1E_3VA
+int refquad_1e_3va_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 1, 2, 7 },
+  { 2, 1, 8 },
+  { 3, 2, 9 },
+  { 3, 4, 10 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_3va_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG_SINGCORNER,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_TRIG,
+  HP_NONE,
+};
+int refquad_1e_3va_newels[][8] =
+{
+  { 1, 7, 5 },
+  { 8, 2, 6 },
+  { 3, 10, 9 },
+  { 7, 8, 6, 5 },
+  { 4, 6, 9, 10 },
+  { 5, 6, 4 }
+};
+HPRef_Struct refquad_1e_3va =
+{
+  HP_QUAD,
+  refquad_1e_3va_splitedges, 
+  0, 0,
+  refquad_1e_3va_newelstypes, 
+  refquad_1e_3va_newels
+};
+
+
+
+
+
+// HP_QUAD_1E_3VB
+int refquad_1e_3vb_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 1, 2, 7 },
+  { 2, 1, 8 },
+  { 4, 1, 9 },
+  { 4, 3, 10 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_3vb_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG_SINGCORNER,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_TRIG,
+  HP_NONE,
+};
+int refquad_1e_3vb_newels[][8] =
+{
+  { 1, 7, 5 },
+  { 8, 2, 6 },
+  { 4, 9, 10 },
+  { 7, 8, 6, 5 },
+  { 5, 3, 10, 9 },
+  { 5, 6, 3 }
+};
+HPRef_Struct refquad_1e_3vb =
+{
+  HP_QUAD,
+  refquad_1e_3vb_splitedges, 
+  0, 0,
+  refquad_1e_3vb_newelstypes, 
+  refquad_1e_3vb_newels
+};
+
+
+
+
+
+// HP_QUAD_1E_3VC
+int refquad_1e_3vc_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 1, 2, 7 },
+  { 3, 2, 8 },
+  { 3, 4, 9 },
+  { 4, 3, 10 },
+  { 4, 1, 11 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_3vc_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGCORNER,
+  HP_TRIG_SINGCORNER,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_1e_3vc_newels[][8] =
+{
+  { 1, 7, 5 },
+  { 3, 9, 8 },
+  { 4, 11, 10 },
+  { 7, 2, 6, 5 },
+  { 5, 6, 8, 11 },
+  { 11, 8, 9, 10 }
+};
+HPRef_Struct refquad_1e_3vc =
+{
+  HP_QUAD,
+  refquad_1e_3vc_splitedges, 
+  0, 0,
+  refquad_1e_3vc_newelstypes, 
+  refquad_1e_3vc_newels
+};
+
+
+
+
+// HP_QUAD_1E_3VD
+int refquad_1e_3vd_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 2, 1, 7 },
+  { 3, 2, 8 },
+  { 3, 4, 9 },
+  { 4, 3, 10 },
+  { 4, 1, 11 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_3vd_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG_SINGCORNER,
+  HP_TRIG_SINGCORNER,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_1e_3vd_newels[][8] =
+{
+  { 7, 2, 6 },
+  { 3, 9, 8 },
+  { 4, 11, 10 },
+  { 1, 7, 6, 5 },
+  { 5, 6, 8, 11 },
+  { 11, 8, 9, 10 }
+};
+HPRef_Struct refquad_1e_3vd =
+{
+  HP_QUAD,
+  refquad_1e_3vd_splitedges, 
+  0, 0,
+  refquad_1e_3vd_newelstypes, 
+  refquad_1e_3vd_newels
+};
+
+
+
+
+
+
+// HP_QUAD_1E_4V
+int refquad_1e_4v_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 1, 2, 7 },
+  { 2, 1, 8 },
+  { 4, 1, 9 },
+  { 3, 2, 10 },
+  { 4, 3, 11 },
+  { 3, 4, 12 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_1e_4v_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG_SINGCORNER,
+  HP_TRIG_SINGCORNER,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_1e_4v_newels[][8] =
+{
+  { 1, 7, 5 },
+  { 8, 2, 6 },
+  { 3, 12, 10 },
+  { 4, 9, 11 },
+  { 7, 8, 6, 5 },
+  { 5, 6, 10, 9 },
+  { 9, 10, 12, 11 }
+};
+HPRef_Struct refquad_1e_4v =
+{
+  HP_QUAD,
+  refquad_1e_4v_splitedges, 
+  0, 0,
+  refquad_1e_4v_newelstypes, 
+  refquad_1e_4v_newels
+};
+
+
+
+
+////////////////////////////////////////////////////////////////////////////////
+
+
+
+// HP_QUAD_2E
+int refquad_2e_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 3, 7 },
+  { 4, 3, 8 },
+  { 0, 0, 0 }
+};
+int refquad_2e_splitfaces[][4] =
+{
+  { 1, 2, 4, 9 },
+  { 0, 0, 0, 0 },
+};
+HPREF_ELEMENT_TYPE refquad_2e_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_2e_newels[][8] =
+{
+  { 1, 5, 9 },
+  { 6, 1, 9 },
+  { 5, 2, 7, 9 },
+  { 4, 6, 9, 8 },
+  { 9, 7, 3, 8 },
+};
+HPRef_Struct refquad_2e =
+{
+  HP_QUAD,
+  refquad_2e_splitedges, 
+  refquad_2e_splitfaces, 
+  0,
+  refquad_2e_newelstypes, 
+  refquad_2e_newels
+};
+
+
+
+
+
+
+
+// HP_QUAD_2E_1VA
+int refquad_2e_1va_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 3, 7 },
+  { 4, 3, 8 },
+  { 2, 1, 10 },
+  { 0, 0, 0 }
+};
+int refquad_2e_1va_splitfaces[][4] =
+{
+  { 1, 2, 4, 9 },
+  { 0, 0, 0, 0 },
+};
+HPREF_ELEMENT_TYPE refquad_2e_1va_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_NONE,
+};
+int refquad_2e_1va_newels[][8] =
+{
+  { 1, 5, 9 },
+  { 6, 1, 9 },
+  { 5, 10, 7, 9 },
+  { 4, 6, 9, 8 },
+  { 9, 7, 3, 8 },
+  { 10, 2, 7 },
+};
+HPRef_Struct refquad_2e_1va =
+{
+  HP_QUAD,
+  refquad_2e_1va_splitedges, 
+  refquad_2e_1va_splitfaces, 
+  0,
+  refquad_2e_1va_newelstypes, 
+  refquad_2e_1va_newels
+};
+
+
+
+
+
+
+// HP_QUAD_2E_1VB
+int refquad_2e_1vb_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 3, 7 },
+  { 4, 3, 8 },
+  { 3, 2, 10 },
+  { 3, 4, 11 },
+  { 0, 0, 0 }
+};
+int refquad_2e_1vb_splitfaces[][4] =
+{
+  { 1, 2, 4, 9 },
+  { 0, 0, 0, 0 },
+};
+HPREF_ELEMENT_TYPE refquad_2e_1vb_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_NONE,
+};
+int refquad_2e_1vb_newels[][8] =
+{
+  { 1, 5, 9 },
+  { 6, 1, 9 },
+  { 5, 2, 7, 9 },
+  { 4, 6, 9, 8 },
+  { 7, 8, 9 },
+  { 8, 7, 10, 11 },
+  { 3, 11, 10 }
+};
+HPRef_Struct refquad_2e_1vb =
+{
+  HP_QUAD,
+  refquad_2e_1vb_splitedges, 
+  refquad_2e_1vb_splitfaces, 
+  0,
+  refquad_2e_1vb_newelstypes, 
+  refquad_2e_1vb_newels
+};
+
+
+
+
+
+// HP_QUAD_2E_1VC
+int refquad_2e_1vc_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 3, 7 },
+  { 4, 1, 8 },
+  { 4, 3, 9 },
+  { 0, 0, 0 }
+};
+int refquad_2e_1vc_splitfaces[][4] =
+{
+  { 1, 2, 4, 10 },
+  { 0, 0, 0, 0 },
+};
+HPREF_ELEMENT_TYPE refquad_2e_1vc_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_2e_1vc_newels[][8] =
+{
+  { 1, 5, 10 },
+  { 6, 1, 10 },
+  { 4, 8, 9 },
+  { 5, 2, 7, 10 },
+  { 8, 6, 10, 9 },
+  { 10, 7, 3, 9 },
+};
+HPRef_Struct refquad_2e_1vc =
+{
+  HP_QUAD,
+  refquad_2e_1vc_splitedges, 
+  refquad_2e_1vc_splitfaces, 
+  0,
+  refquad_2e_1vc_newelstypes, 
+  refquad_2e_1vc_newels
+};
+
+
+
+
+
+
+
+
+
+
+// HP_QUAD_2E
+int refquad_2e_2va_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 3, 7 },
+  { 4, 3, 8 },
+  { 3, 2, 10 },
+  { 3, 4, 11 },
+  { 2, 1, 12 },
+  { 0, 0, 0 }
+};
+int refquad_2e_2va_splitfaces[][4] =
+{
+  { 1, 2, 4, 9 },
+  { 0, 0, 0, 0 },
+};
+HPREF_ELEMENT_TYPE refquad_2e_2va_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_NONE,
+};
+int refquad_2e_2va_newels[][8] =
+{
+  { 1, 5, 9 },
+  { 6, 1, 9 },
+  { 5, 12, 7, 9 },
+  { 4, 6, 9, 8 },
+  { 7, 8, 9 },
+  { 8, 7, 10, 11 },
+  { 3, 11, 10 },
+  { 12, 2, 7 }
+};
+HPRef_Struct refquad_2e_2va =
+{
+  HP_QUAD,
+  refquad_2e_2va_splitedges, 
+  refquad_2e_2va_splitfaces, 
+  0,
+  refquad_2e_2va_newelstypes, 
+  refquad_2e_2va_newels
+};
+
+
+
+
+
+
+// HP_QUAD_2E_2VB
+int refquad_2e_2vb_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 1, 7 },
+  { 2, 3, 8 },
+  { 4, 1, 9 },
+  { 4, 3, 10 },
+  { 0, 0, 0 }
+};
+int refquad_2e_2vb_splitfaces[][4] =
+{
+  { 1, 2, 4, 11 },
+  { 0, 0, 0, 0 },
+};
+HPREF_ELEMENT_TYPE refquad_2e_2vb_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_2e_2vb_newels[][8] =
+{
+  { 1, 5, 11 },
+  { 6, 1, 11 },
+  { 4, 9, 10 },
+  { 7, 2, 8 },
+  { 5, 7, 8, 11 },
+  { 9, 6, 11, 10 },
+  { 3, 10, 11, 8 },
+};
+HPRef_Struct refquad_2e_2vb =
+{
+  HP_QUAD,
+  refquad_2e_2vb_splitedges, 
+  refquad_2e_2vb_splitfaces, 
+  0,
+  refquad_2e_2vb_newelstypes, 
+  refquad_2e_2vb_newels
+};
+
+
+
+
+
+
+
+
+
+
+// HP_QUAD_2E
+int refquad_2e_2vc_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 3, 7 },
+  { 4, 3, 8 },
+  { 3, 2, 10 },
+  { 3, 4, 11 },
+  { 4, 1, 12 },
+  { 0, 0, 0 }
+};
+int refquad_2e_2vc_splitfaces[][4] =
+{
+  { 1, 2, 4, 9 },
+  { 0, 0, 0, 0 },
+};
+HPREF_ELEMENT_TYPE refquad_2e_2vc_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_NONE,
+};
+int refquad_2e_2vc_newels[][8] =
+{
+  { 1, 5, 9 },
+  { 6, 1, 9 },
+  { 5, 2, 7, 9 },
+  { 12, 6, 9, 8 },
+  { 7, 8, 9 },
+  { 8, 7, 10, 11 },
+  { 3, 11, 10 },
+  { 4, 12, 8 }
+};
+HPRef_Struct refquad_2e_2vc =
+{
+  HP_QUAD,
+  refquad_2e_2vc_splitedges, 
+  refquad_2e_2vc_splitfaces, 
+  0,
+  refquad_2e_2vc_newelstypes, 
+  refquad_2e_2vc_newels
+};
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+// HP_QUAD_2E
+int refquad_2e_3v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 3, 7 },
+  { 4, 3, 8 },
+  { 3, 2, 10 },
+  { 3, 4, 11 },
+  { 2, 1, 12 },
+  { 4, 1, 13 },
+  { 0, 0, 0 }
+};
+int refquad_2e_3v_splitfaces[][4] =
+{
+  { 1, 2, 4, 9 },
+  { 0, 0, 0, 0 },
+};
+HPREF_ELEMENT_TYPE refquad_2e_3v_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_NONE,
+};
+int refquad_2e_3v_newels[][8] =
+{
+  { 1, 5, 9 },
+  { 6, 1, 9 },
+  { 5, 12, 7, 9 },
+  { 13, 6, 9, 8 },
+  { 7, 8, 9 },
+  { 8, 7, 10, 11 },
+  { 3, 11, 10 },
+  { 12, 2, 7 },
+  { 4, 13, 8 }
+};
+HPRef_Struct refquad_2e_3v =
+{
+  HP_QUAD,
+  refquad_2e_3v_splitedges, 
+  refquad_2e_3v_splitfaces, 
+  0,
+  refquad_2e_3v_newelstypes, 
+  refquad_2e_3v_newels
+};
+
+
+
+
+
+
+
+
+
+
+// HP_QUAD_2EB_0V
+int refquad_2eb_0v_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 3, 2, 7 },
+  { 4, 1, 8 },
+  { 0, 0, 0 }
+};
+int refquad_2eb_0v_splitfaces[][4] =
+{
+  { 0, 0, 0, 0 },
+};
+HPREF_ELEMENT_TYPE refquad_2eb_0v_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_2eb_0v_newels[][8] =
+{
+  { 1, 2, 6, 5 },
+  { 3, 4, 8, 7 },
+  { 5, 6, 7, 8 }
+};
+HPRef_Struct refquad_2eb_0v =
+{
+  HP_QUAD,
+  refquad_2eb_0v_splitedges, 
+  refquad_2eb_0v_splitfaces, 
+  0,
+  refquad_2eb_0v_newelstypes, 
+  refquad_2eb_0v_newels
+};
+
+
+
+
+
+
+
+
+// HP_QUAD_2EB_1VA
+int refquad_2eb_1va_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 3, 2, 7 },
+  { 4, 1, 8 },
+  { 1, 2, 9 },
+  { 0, 0, 0 }
+};
+int refquad_2eb_1va_splitfaces[][4] =
+{
+  { 0, 0, 0, 0 },
+};
+HPREF_ELEMENT_TYPE refquad_2eb_1va_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_2eb_1va_newels[][8] =
+{
+  { 9, 2, 6, 5 },
+  { 3, 4, 8, 7 },
+  { 1, 9, 5 },
+  { 5, 6, 7, 8 }
+};
+HPRef_Struct refquad_2eb_1va =
+{
+  HP_QUAD,
+  refquad_2eb_1va_splitedges, 
+  refquad_2eb_1va_splitfaces, 
+  0,
+  refquad_2eb_1va_newelstypes, 
+  refquad_2eb_1va_newels
+};
+
+
+
+
+// HP_QUAD_2EB_1VB
+int refquad_2eb_1vb_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 3, 2, 7 },
+  { 4, 1, 8 },
+  { 2, 1, 9 },
+  { 0, 0, 0 }
+};
+int refquad_2eb_1vb_splitfaces[][4] =
+{
+  { 0, 0, 0, 0 },
+};
+HPREF_ELEMENT_TYPE refquad_2eb_1vb_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_2eb_1vb_newels[][8] =
+{
+  { 1, 9, 6, 5 },
+  { 3, 4, 8, 7 },
+  { 9, 2, 6 },
+  { 5, 6, 7, 8 }
+};
+HPRef_Struct refquad_2eb_1vb =
+{
+  HP_QUAD,
+  refquad_2eb_1vb_splitedges, 
+  refquad_2eb_1vb_splitfaces, 
+  0,
+  refquad_2eb_1vb_newelstypes, 
+  refquad_2eb_1vb_newels
+};
+
+
+
+
+
+
+// HP_QUAD_2EB_2VA
+int refquad_2eb_2va_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 3, 2, 7 },
+  { 4, 1, 8 },
+  { 1, 2, 9 },
+  { 2, 1, 10 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_2eb_2va_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_2eb_2va_newels[][8] =
+{
+  { 9, 10, 6, 5 },
+  { 3, 4, 8, 7 },
+  { 1, 9, 5 },
+  { 10, 2, 6 },
+  { 5, 6, 7, 8 }
+};
+HPRef_Struct refquad_2eb_2va =
+{
+  HP_QUAD,
+  refquad_2eb_2va_splitedges, 
+  0, 0,
+  refquad_2eb_2va_newelstypes, 
+  refquad_2eb_2va_newels
+};
+
+
+
+// HP_QUAD_2EB_2VB
+int refquad_2eb_2vb_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 3, 2, 7 },
+  { 4, 1, 8 },
+  { 1, 2, 9 },
+  { 3, 4, 10 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_2eb_2vb_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_2eb_2vb_newels[][8] =
+{
+  { 9, 2, 6, 5 },
+  { 10, 4, 8, 7 },
+  { 1, 9, 5 },
+  { 3, 10, 7 },
+  { 5, 6, 7, 8 }
+};
+HPRef_Struct refquad_2eb_2vb =
+{
+  HP_QUAD,
+  refquad_2eb_2vb_splitedges, 
+  0, 0,
+  refquad_2eb_2vb_newelstypes, 
+  refquad_2eb_2vb_newels
+};
+
+
+
+// HP_QUAD_2EB_2VC
+int refquad_2eb_2vc_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 3, 2, 7 },
+  { 4, 1, 8 },
+  { 1, 2, 9 },
+  { 4, 3, 10 },
+  { 0, 0, 0 }
+};
+int refquad_2eb_2vc_splitfaces[][4] =
+{
+  { 0, 0, 0, 0 },
+};
+HPREF_ELEMENT_TYPE refquad_2eb_2vc_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_2eb_2vc_newels[][8] =
+{
+  { 9, 2, 6, 5 },
+  { 3, 10, 8, 7 },
+  { 1, 9, 5 },
+  { 10, 4, 8 },
+  { 5, 6, 7, 8 }
+};
+HPRef_Struct refquad_2eb_2vc =
+{
+  HP_QUAD,
+  refquad_2eb_2vc_splitedges, 
+  refquad_2eb_2vc_splitfaces, 
+  0,
+  refquad_2eb_2vc_newelstypes, 
+  refquad_2eb_2vc_newels
+};
+
+
+
+
+// HP_QUAD_2EB_2VD
+int refquad_2eb_2vd_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 3, 2, 7 },
+  { 4, 1, 8 },
+  { 2, 1, 9 },
+  { 4, 3, 10 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_2eb_2vd_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_2eb_2vd_newels[][8] =
+{
+  { 1, 9, 6, 5 },
+  { 3, 10, 8, 7 },
+  { 9, 2, 6 },
+  { 10, 4, 8 },
+  { 5, 6, 7, 8 }
+};
+HPRef_Struct refquad_2eb_2vd =
+{
+  HP_QUAD,
+  refquad_2eb_2vd_splitedges, 
+  0, 0,
+  refquad_2eb_2vd_newelstypes, 
+  refquad_2eb_2vd_newels
+};
+
+
+
+
+
+
+
+// HP_QUAD_2EB_3VA
+int refquad_2eb_3va_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 1, 2, 7 },
+  { 2, 1, 8 },
+  { 3, 2, 9 },
+  { 4, 1, 10 },
+  { 3, 4, 11 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_2eb_3va_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_2eb_3va_newels[][8] =
+{
+  { 1, 7, 5 },
+  { 8, 2, 6 },
+  { 3, 11, 9},
+  { 7, 8, 6, 5 },
+  { 11, 4, 10, 9 },
+  { 5, 6, 9, 10 }
+};
+HPRef_Struct refquad_2eb_3va =
+{
+  HP_QUAD,
+  refquad_2eb_3va_splitedges, 
+  0, 0,
+  refquad_2eb_3va_newelstypes, 
+  refquad_2eb_3va_newels
+};
+
+
+
+
+
+// HP_QUAD_2EB_3VB
+int refquad_2eb_3vb_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 1, 2, 7 },
+  { 2, 1, 8 },
+  { 3, 2, 9 },
+  { 4, 1, 10 },
+  { 4, 3, 11 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE refquad_2eb_3vb_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_2eb_3vb_newels[][8] =
+{
+  { 1, 7, 5 },
+  { 8, 2, 6 },
+  { 11, 4, 10 },
+  { 7, 8, 6, 5 },
+  { 3, 11, 10, 9 },
+  { 5, 6, 9, 10 }
+};
+HPRef_Struct refquad_2eb_3vb =
+{
+  HP_QUAD,
+  refquad_2eb_3vb_splitedges, 
+  0, 0,
+  refquad_2eb_3vb_newelstypes, 
+  refquad_2eb_3vb_newels
+};
+
+
+
+
+
+
+// HP_QUAD_2EB_4V
+int refquad_2eb_4v_splitedges[][3] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 6 },
+  { 3, 2, 7 },
+  { 4, 1, 8 },
+  { 1, 2, 9 },
+  { 2, 1, 10 },
+  { 3, 4, 11 },
+  { 4, 3, 12 },
+  { 0, 0, 0 }
+};
+int refquad_2eb_4v_splitfaces[][4] =
+{
+  { 0, 0, 0, 0 },
+};
+HPREF_ELEMENT_TYPE refquad_2eb_4v_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_NONE,
+};
+int refquad_2eb_4v_newels[][8] =
+{
+  { 9, 10, 6, 5 },
+  { 11, 12, 8, 7 },
+  { 5, 6, 7, 8 },
+  { 1, 9, 5 },
+  { 10, 2, 6 },
+  { 3, 11, 7 },
+  { 12, 4, 8 },
+};
+HPRef_Struct refquad_2eb_4v =
+{
+  HP_QUAD,
+  refquad_2eb_4v_splitedges, 
+  refquad_2eb_4v_splitfaces, 
+  0,
+  refquad_2eb_4v_newelstypes, 
+  refquad_2eb_4v_newels
+};
+
+
+
+
+
+
+
+// HP_QUAD_3E
+int refquad_3e_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 1, 7 },
+  { 2, 3, 8 },
+  { 3, 4, 10 },
+  { 4, 3, 12 },
+  { 0, 0, 0 }
+};
+
+int refquad_3e_splitfaces[][4] =
+{
+  { 1, 2, 4, 13 },
+  { 2, 3, 1, 14 },
+  { 0, 0, 0, 0 },
+};
+
+HPREF_ELEMENT_TYPE refquad_3e_newelstypes[] =
+{
+  HP_QUAD_2E,
+  HP_QUAD_2E,
+//   HP_TRIG_SINGEDGECORNER1,
+//   HP_TRIG_SINGEDGECORNER2,
+//   HP_TRIG_SINGEDGECORNER2,
+//   HP_TRIG_SINGEDGECORNER1,
+
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_3e_newels[][8] =
+{
+//   { 1, 5, 13 },
+//   { 6, 1, 13 },
+//   { 7, 2, 14 },
+//   { 2, 8, 14 },
+  { 1, 5, 13, 6 },
+  { 2, 8, 14, 7 },
+  { 5, 7, 14, 13 },
+  { 8, 3, 10, 14 },
+  { 4, 6, 13, 12 },
+  { 13, 14, 10, 12 }
+};
+HPRef_Struct refquad_3e =
+{
+  HP_QUAD,
+  refquad_3e_splitedges, 
+  refquad_3e_splitfaces, 
+  0,
+  refquad_3e_newelstypes, 
+  refquad_3e_newels
+};
+
+
+
+
+
+
+
+// HP_QUAD_3E_3VA
+int refquad_3e_3va_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 1, 7 },
+  { 2, 3, 8 },
+  { 3, 4, 10 },
+  { 3, 2, 11 },
+  { 4, 3, 12 },
+  { 0, 0, 0 }
+};
+
+int refquad_3e_3va_splitfaces[][4] =
+{
+  { 1, 2, 4, 13 },
+  { 2, 3, 1, 14 },
+  { 0, 0, 0, 0 },
+};
+
+HPREF_ELEMENT_TYPE refquad_3e_3va_newelstypes[] =
+{
+  HP_QUAD_2E,
+  HP_QUAD_2E,
+
+//   HP_TRIG_SINGEDGECORNER1,
+//   HP_TRIG_SINGEDGECORNER2,
+//   HP_TRIG_SINGEDGECORNER2,
+//   HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_3e_3va_newels[][8] =
+{
+//   { 1, 5, 13 },
+//   { 6, 1, 13 },
+//   { 7, 2, 14 },
+//   { 2, 8, 14 },
+  { 1, 5, 13, 6 },
+  { 2, 8, 14, 7 },
+  { 11, 3, 10 },
+  { 5, 7, 14, 13 },
+  { 8, 11, 10, 14 },
+  { 4, 6, 13, 12 },
+  { 13, 14, 10, 12 }
+};
+HPRef_Struct refquad_3e_3va =
+{
+  HP_QUAD,
+  refquad_3e_3va_splitedges, 
+  refquad_3e_3va_splitfaces, 
+  0,
+  refquad_3e_3va_newelstypes, 
+  refquad_3e_3va_newels
+};
+
+
+
+
+
+
+
+
+
+
+// HP_QUAD_3E_3VB
+int refquad_3e_3vb_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 1, 7 },
+  { 2, 3, 8 },
+  { 3, 4, 10 },
+  { 4, 1, 11 },
+  { 4, 3, 12 },
+  { 0, 0, 0 }
+};
+
+int refquad_3e_3vb_splitfaces[][4] =
+{
+  { 1, 2, 4, 13 },
+  { 2, 3, 1, 14 },
+  { 0, 0, 0, 0 },
+};
+
+HPREF_ELEMENT_TYPE refquad_3e_3vb_newelstypes[] =
+{
+  HP_QUAD_2E,
+  HP_QUAD_2E,
+
+//   HP_TRIG_SINGEDGECORNER1,
+//   HP_TRIG_SINGEDGECORNER2,
+//   HP_TRIG_SINGEDGECORNER2,
+//   HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER1,
+
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_3e_3vb_newels[][8] =
+{
+//   { 1, 5, 13 },
+//   { 6, 1, 13 },
+//   { 7, 2, 14 },
+//   { 2, 8, 14 },
+  { 1, 5, 13, 6 },
+  { 2, 8, 14, 7 },
+  { 4, 11, 12 },
+  { 5, 7, 14, 13 },
+  { 8, 3, 10, 14 },
+  { 11, 6, 13, 12 },
+  { 13, 14, 10, 12 }
+};
+HPRef_Struct refquad_3e_3vb =
+{
+  HP_QUAD,
+  refquad_3e_3vb_splitedges, 
+  refquad_3e_3vb_splitfaces, 
+  0,
+  refquad_3e_3vb_newelstypes, 
+  refquad_3e_3vb_newels
+};
+
+
+
+
+
+
+
+
+
+// HP_QUAD_3E_4V
+int refquad_3e_4v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 1, 7 },
+  { 2, 3, 8 },
+  { 3, 4, 10 },
+  { 3, 2, 11 },
+  { 4, 3, 12 },
+  { 4, 1, 15 },
+  { 0, 0, 0 }
+};
+
+int refquad_3e_4v_splitfaces[][4] =
+{
+  { 1, 2, 4, 13 },
+  { 2, 3, 1, 14 },
+  { 0, 0, 0, 0 },
+};
+
+HPREF_ELEMENT_TYPE refquad_3e_4v_newelstypes[] =
+{
+  HP_QUAD_2E,
+  HP_QUAD_2E,
+
+//   HP_TRIG_SINGEDGECORNER1,
+//   HP_TRIG_SINGEDGECORNER2,
+//   HP_TRIG_SINGEDGECORNER2,
+//   HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG_SINGEDGECORNER1,
+
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_3e_4v_newels[][8] =
+{
+//   { 1, 5, 13 },
+//   { 6, 1, 13 },
+//   { 7, 2, 14 },
+//   { 2, 8, 14 },
+  { 1, 5, 13, 6 },
+  { 2, 8, 14, 7 },
+  { 11, 3, 10 },
+  { 4, 15, 12 },
+  { 5, 7, 14, 13 },
+  { 8, 11, 10, 14 },
+  { 15, 6, 13, 12 },
+  { 13, 14, 10, 12 }
+};
+HPRef_Struct refquad_3e_4v =
+{
+  HP_QUAD,
+  refquad_3e_4v_splitedges, 
+  refquad_3e_4v_splitfaces, 
+  0,
+  refquad_3e_4v_newelstypes, 
+  refquad_3e_4v_newels
+};
+
+
+
+
+
+
+
+
+
+// HP_QUAD_4E
+int refquad_4e_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 4, 6 },
+  { 2, 1, 7 },
+  { 2, 3, 8 },
+  { 3, 2, 9 },
+  { 3, 4, 10 },
+  { 4, 1, 11 },
+  { 4, 3, 12 },
+  { 0, 0, 0 }
+};
+
+int refquad_4e_splitfaces[][4] =
+{
+  { 1, 2, 4, 13 },
+  { 2, 3, 1, 14 },
+  { 3, 4, 2, 15 },
+  { 4, 1, 3, 16 },
+  { 0, 0, 0, 0 },
+};
+
+HPREF_ELEMENT_TYPE refquad_4e_newelstypes[] =
+{
+  HP_QUAD_2E,
+  HP_QUAD_2E,
+  HP_QUAD_2E,
+  HP_QUAD_2E,
+
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+
+  HP_QUAD,
+  HP_NONE,
+};
+int refquad_4e_newels[][8] =
+{
+  { 1, 5, 13, 6 },
+  { 2, 8, 14, 7 },
+  { 3, 10, 15, 9 },
+  { 4, 11, 16, 12 },
+  { 5, 7, 14, 13 },
+  { 8, 9, 15, 14 },
+  { 10, 12, 16, 15 },
+  { 11, 6, 13, 16 },
+  { 13, 14, 15, 16 }
+};
+HPRef_Struct refquad_4e =
+{
+  HP_QUAD,
+  refquad_4e_splitedges, 
+  refquad_4e_splitfaces, 
+  0,
+  refquad_4e_newelstypes, 
+  refquad_4e_newels
+};
+
+
diff --git a/Netgen/libsrc/meshing/hpref_tet.hpp b/Netgen/libsrc/meshing/hpref_tet.hpp
new file mode 100644
index 0000000000..b071269ec8
--- /dev/null
+++ b/Netgen/libsrc/meshing/hpref_tet.hpp
@@ -0,0 +1,2842 @@
+
+
+// HP_TET
+int reftet_splitedges[][3] =
+{
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftet_newelstypes[] =
+{
+  HP_TET,
+  HP_NONE,
+};
+int reftet_newels[][8] =
+{
+  { 1, 2, 3, 4 },
+};
+HPRef_Struct reftet =
+{
+  HP_TET,
+  reftet_splitedges, 
+  0, 0,
+  reftet_newelstypes, 
+  reftet_newels
+};
+
+
+
+/* *********** Tet - Refinement - 0 edges *************** */
+
+// HP_TET_0E_1V
+int reftet_0e_1v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftet_0e_1v_newelstypes[] =
+{
+  HP_TET_0E_1V,
+  HP_PRISM,
+  HP_NONE,
+};
+int reftet_0e_1v_newels[][8] =
+{
+  { 1, 5, 6, 7 },
+  { 5, 6, 7, 2, 3, 4 }
+};
+HPRef_Struct reftet_0e_1v =
+{
+  HP_TET,
+  reftet_0e_1v_splitedges, 
+  0, 0,
+  reftet_0e_1v_newelstypes, 
+  reftet_0e_1v_newels
+};
+
+
+
+// HP_TET_0E_2V
+int reftet_0e_2v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftet_0e_2v_newelstypes[] =
+{
+  HP_TET_0E_1V,
+  HP_TET_0E_1V,
+  HP_PRISM,
+  HP_PRISM,
+  HP_NONE,
+};
+int reftet_0e_2v_newels[][8] =
+{
+  { 1, 5, 6, 7 },
+  { 2, 10, 9, 8 },
+  { 5, 6, 7, 8, 9, 10 },
+  { 4, 10, 7, 3, 9, 6 },
+};
+HPRef_Struct reftet_0e_2v =
+{
+  HP_TET,
+  reftet_0e_2v_splitedges, 
+  0, 0,
+  reftet_0e_2v_newelstypes, 
+  reftet_0e_2v_newels
+};
+
+
+
+
+
+// HP_TET_0E_3V
+int reftet_0e_3v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 0, 0, 0 }
+};
+int reftet_0e_3v_splitfaces[][4] =
+  {
+    { 1, 2, 3, 14 },
+    { 2, 3, 1, 15 },
+    { 3, 1, 2, 16 },
+    { 0, 0, 0, 0 },
+  };
+HPREF_ELEMENT_TYPE reftet_0e_3v_newelstypes[] =
+{
+  HP_PYRAMID_0E_1V,
+  HP_PYRAMID_0E_1V,
+  HP_PYRAMID_0E_1V,
+  HP_PRISM,
+  HP_PRISM,
+  HP_PRISM,
+  HP_PRISM,
+  HP_TET,
+  HP_NONE,
+};
+int reftet_0e_3v_newels[][8] =
+{
+  { 1, 5, 14, 6, 7 },
+  { 2, 9, 15, 8, 10 },
+  { 3, 11, 16, 12, 13 },
+  { 5, 14, 7, 8, 15, 10 },
+  { 9, 15, 10, 12, 16, 13 },
+  { 6, 7, 14, 11, 13, 16 },
+  { 14, 15, 16, 7, 10, 13 },
+  { 7, 10, 13, 4 }
+};
+HPRef_Struct reftet_0e_3v =
+{
+  HP_TET,
+  reftet_0e_3v_splitedges, 
+  reftet_0e_3v_splitfaces, 
+  0,
+  reftet_0e_3v_newelstypes, 
+  reftet_0e_3v_newels
+};
+
+
+
+
+
+// HP_TET_0E_4V
+int reftet_0e_4v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_0e_4v_splitfaces[][4] =
+  {
+    { 1, 2, 3, 17 },
+    { 1, 2, 4, 18 },
+    { 1, 3, 4, 19 },
+
+    { 2, 1, 3, 20 },
+    { 2, 1, 4, 21 },
+    { 2, 3, 4, 22 },
+
+    { 3, 1, 2, 23 },
+    { 3, 1, 4, 24 },
+    { 3, 2, 4, 25 },
+
+    { 4, 1, 2, 26 },
+    { 4, 1, 3, 27 },
+    { 4, 2, 3, 28 },
+    { 0, 0, 0, 0 },
+  };
+int reftet_0e_4v_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 29 },
+    { 2, 3, 4, 1, 30 },
+    { 3, 4, 1, 2, 31 },
+    { 4, 1, 2, 3, 32 },
+    { 0 },
+  };
+HPREF_ELEMENT_TYPE reftet_0e_4v_newelstypes[] =
+{
+  HP_HEX_0E_1V,
+  HP_HEX_0E_1V,
+  HP_HEX_0E_1V,
+  HP_HEX_0E_1V,
+  HP_PRISM, HP_PRISM, 
+  HP_PRISM, HP_PRISM, 
+  HP_PRISM, HP_PRISM, 
+  HP_PRISM, HP_PRISM, 
+  HP_PRISM, HP_PRISM, 
+  HP_PRISM, HP_PRISM, 
+  /*
+  HP_HEX, 
+  HP_HEX, 
+  HP_HEX, 
+  HP_HEX, 
+  HP_HEX, 
+  HP_HEX, 
+  */
+  HP_PRISM,
+  HP_PRISM,
+  HP_PRISM,
+  HP_PRISM,
+  HP_TET,
+  HP_NONE,
+};
+int reftet_0e_4v_newels[][8] =
+{
+  { 1, 5, 17, 6, 7, 18, 29, 19 },
+  { 2, 9, 20, 8, 10, 22, 30, 21 },
+  { 3, 11, 23, 12, 13, 24, 31, 25 },
+  { 4, 15, 26, 14, 16, 28, 32, 27 },
+  { 5, 17, 18, 8, 20, 21 },
+  { 18, 17, 29, 21, 20, 30 },
+  { 6, 19, 17,  11, 24, 23 },
+  { 17, 19, 29,  23, 24, 31 },
+  { 7, 18, 19, 14, 26, 27 },
+  { 19, 18, 29, 27, 26, 32 },
+  { 9, 20, 22, 12, 23, 25 },
+  { 22, 20, 30, 25, 23, 31 },
+  { 10, 22, 21, 15, 28, 26 },
+  { 21, 22, 30, 26, 28, 32 },
+  { 13, 24, 25, 16, 27, 28 },
+  { 25, 24, 31, 28, 27, 32 },
+  /*
+  { 5, 17, 29, 18, 8, 20, 30, 21 },
+  { 6, 19, 29, 17, 11, 24, 31, 23 },
+  { 7, 18, 29, 19, 14, 26, 32, 27 },
+  { 9, 20, 30, 22, 12, 23, 31, 25 },
+  { 10, 22, 30, 21, 15, 28, 32, 26 },
+  { 13, 24, 31, 25, 16, 27, 32, 28 },
+  */
+  { 17, 20, 23, 29, 30, 31 },
+  { 18, 26, 21, 29, 32, 30 },
+  { 19, 24, 27, 29, 31, 32 },
+  { 22, 28, 25, 30, 32, 31 },
+
+  { 29, 30, 31, 32 },
+};
+HPRef_Struct reftet_0e_4v =
+{
+  HP_TET,
+  reftet_0e_4v_splitedges, 
+  reftet_0e_4v_splitfaces, 
+  reftet_0e_4v_splitelements, 
+  reftet_0e_4v_newelstypes, 
+  reftet_0e_4v_newels
+};
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+/* *********** Tet - Refinement - 1 edge *************** */
+
+
+
+// HP_TET_1E_0V
+int reftet_1e_0v_splitedges[][3] =
+{
+  { 1, 3, 5 },
+  { 1, 4, 6 },
+  { 2, 3, 7 },
+  { 2, 4, 8 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftet_1e_0v_newelstypes[] =
+{
+  HP_PRISM_SINGEDGE,
+  HP_PRISM,
+  HP_NONE,
+};
+int reftet_1e_0v_newels[][8] =
+{
+  { 1, 5, 6, 2, 7, 8 },
+  { 7, 3, 5, 8, 4, 6 }
+};
+HPRef_Struct reftet_1e_0v =
+{
+  HP_TET,
+  reftet_1e_0v_splitedges, 
+  0, 0,
+  reftet_1e_0v_newelstypes, 
+  reftet_1e_0v_newels
+};
+
+
+
+
+
+// HP_TET_1E_1VA
+int reftet_1e_1va_splitedges[][3] =
+{
+  { 1, 3, 5 },
+  { 1, 4, 6 },
+  { 2, 3, 7 },
+  { 2, 4, 8 },
+  { 1, 2, 9 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftet_1e_1va_newelstypes[] =
+{
+  HP_TET_1E_1VA,
+  HP_PRISM_SINGEDGE,
+  HP_PRISM,
+  HP_NONE,
+};
+int reftet_1e_1va_newels[][8] =
+{
+  { 1, 9, 5, 6 },
+  { 9, 5, 6, 2, 7, 8 },
+  { 7, 3, 5, 8, 4, 6 }
+};
+HPRef_Struct reftet_1e_1va =
+{
+  HP_TET,
+  reftet_1e_1va_splitedges, 
+  0, 0,
+  reftet_1e_1va_newelstypes, 
+  reftet_1e_1va_newels
+};
+
+
+
+
+
+
+// HP_TET_1E_1VB
+int reftet_1e_1vb_splitedges[][3] =
+{
+  { 1, 3, 5 },
+  { 1, 4, 6 },
+  { 2, 3, 7 },
+  { 2, 4, 8 },
+  { 4, 1, 9 },
+  { 4, 2, 10 },
+  { 4, 3, 11 },
+  { 0, 0, 0 }
+};
+int reftet_1e_1vb_splitelements[][5] =
+{
+  { 4, 1, 2, 3, 12 },
+  { 0 }
+};
+
+HPREF_ELEMENT_TYPE reftet_1e_1vb_newelstypes[] =
+{
+  HP_PRISM_SINGEDGE,
+  HP_TET_0E_1V,
+  HP_PYRAMID,
+  HP_TET,
+  HP_PYRAMID, 
+  HP_TET,
+  HP_PYRAMID,
+  HP_TET,
+  HP_PYRAMID,
+  HP_TET,
+  HP_NONE,
+};
+int reftet_1e_1vb_newels[][8] =
+{
+  { 1, 5, 6, 2, 7, 8 },
+  { 4, 11, 10, 9 },
+  { 7, 8, 10, 11, 12 },
+  { 3, 7, 11, 12 },
+  { 5, 11, 9, 6, 12 },
+  { 5, 3, 11, 12 },
+  { 6, 9, 10, 8, 12 },
+  { 5, 7, 3, 12 },
+  { 5, 6, 8, 7, 12 },
+  { 9, 11, 10, 12 }
+};
+HPRef_Struct reftet_1e_1vb =
+{
+  HP_TET,
+  reftet_1e_1vb_splitedges, 
+  0,
+  reftet_1e_1vb_splitelements, 
+  reftet_1e_1vb_newelstypes, 
+  reftet_1e_1vb_newels
+};
+
+
+
+
+
+
+
+
+// HP_TET_1E_2VA
+int reftet_1e_2va_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftet_1e_2va_newelstypes[] =
+{
+  HP_TET_1E_1VA,
+  HP_TET_1E_1VA,
+  HP_PRISM_SINGEDGE,
+  HP_PRISM,
+  HP_NONE,
+};
+int reftet_1e_2va_newels[][8] =
+{
+  { 1, 5, 6, 7 },
+  { 2, 8, 10, 9 },
+  { 5, 6, 7, 8, 9, 10 },
+  { 4, 10, 7, 3, 9, 6 },
+};
+HPRef_Struct reftet_1e_2va =
+{
+  HP_TET,
+  reftet_1e_2va_splitedges, 
+  0, 0,
+  reftet_1e_2va_newelstypes, 
+  reftet_1e_2va_newels
+};
+
+
+
+
+
+
+
+// HP_TET_1E_2VB
+int reftet_1e_2vb_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 3, 8 },
+  { 2, 4, 9 },
+  { 3, 1, 10 },
+  { 3, 2, 11 },
+  { 3, 4, 12 },
+  { 0, 0, 0 }
+};
+int reftet_1e_2vb_splitelements[][5] =
+{
+  { 3, 4, 1, 2, 13 },
+  { 0 }
+};
+
+HPREF_ELEMENT_TYPE reftet_1e_2vb_newelstypes[] =
+{
+  HP_TET_1E_1VA,
+  HP_PRISM_SINGEDGE,
+  HP_TET_0E_1V,
+  HP_PYRAMID,
+  HP_TET,
+  HP_PYRAMID, 
+  HP_TET,
+  HP_PYRAMID,
+  HP_TET,
+  HP_PYRAMID,
+  HP_TET,
+  HP_NONE,
+};
+int reftet_1e_2vb_newels[][8] =
+{
+  { 1, 5, 6, 7 },
+  { 5, 6, 7, 2, 8, 9 },
+  { 3, 10, 11, 12 },
+
+  { 8, 9, 12, 11, 13 },
+  { 4, 12, 9, 13 },
+  { 6, 10, 12, 7, 13 },
+  { 4, 7, 12, 13 },
+  { 6, 8, 11, 10, 13 },
+  { 4, 9, 7, 13 },
+  { 6, 7, 9, 8, 13 },
+  { 10, 11, 12, 13 },
+};
+HPRef_Struct reftet_1e_2vb =
+{
+  HP_TET,
+  reftet_1e_2vb_splitedges, 
+  0,
+  reftet_1e_2vb_splitelements, 
+  reftet_1e_2vb_newelstypes, 
+  reftet_1e_2vb_newels
+};
+
+
+
+
+
+
+// HP_TET_1E_2VC
+int reftet_1e_2vc_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 3, 8 },
+  { 2, 4, 9 },
+  { 4, 1, 10 },
+  { 4, 2, 11 },
+  { 4, 3, 12 },
+  { 0, 0, 0 }
+};
+int reftet_1e_2vc_splitelements[][5] =
+{
+  { 4, 1, 2, 3, 13 },
+  { 0 }
+};
+
+HPREF_ELEMENT_TYPE reftet_1e_2vc_newelstypes[] =
+{
+  HP_TET_1E_1VA,
+  HP_PRISM_SINGEDGE,
+  HP_TET_0E_1V,
+  HP_PYRAMID,
+  HP_TET,
+  HP_PYRAMID, 
+  HP_TET,
+  HP_PYRAMID,
+  HP_TET,
+  HP_PYRAMID,
+  HP_TET,
+  HP_NONE,
+};
+int reftet_1e_2vc_newels[][8] =
+{
+  { 1, 5, 6, 7 },
+  { 5, 6, 7, 2, 8, 9 },
+  { 4, 11, 10, 12 },
+  { 8, 9, 11, 12, 13 },
+  { 3, 8, 12, 13 },
+  { 7, 6, 12, 10, 13 },
+  { 3, 12, 6, 13 },
+  { 9, 7, 10, 11, 13 },
+  { 3, 6, 8, 13 },
+  { 6, 7, 9, 8, 13 },
+  { 10, 12, 11, 13 }
+};
+HPRef_Struct reftet_1e_2vc =
+{
+  HP_TET,
+  reftet_1e_2vc_splitedges, 
+  0,
+  reftet_1e_2vc_splitelements, 
+  reftet_1e_2vc_newelstypes, 
+  reftet_1e_2vc_newels
+};
+
+
+
+
+
+
+
+
+/*
+
+// HP_TET_1E_2VD
+int reftet_1e_2vd_splitedges[][3] =
+{
+  { 1, 3, 5 },
+  { 1, 4, 6 },
+  { 2, 3, 7 },
+  { 2, 4, 8 },
+  { 3, 1, 9 },
+  { 3, 2, 10 },
+  { 3, 4, 11 },
+  { 4, 1, 12 },
+  { 4, 2, 13 },
+  { 4, 3, 14 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftet_1e_2vd_newelstypes[] =
+{
+  HP_PRISM_SINGEDGE,
+  HP_TET_0E_1V,
+  HP_TET_0E_1V,
+  HP_PRISM,
+  HP_HEX,
+  HP_NONE,
+};
+int reftet_1e_2vd_newels[][8] =
+{
+  { 1, 5, 6, 2, 7, 8 },
+  { 4, 13, 12, 14 },
+  { 3, 10, 11, 9 },
+  { 14, 13, 12, 11, 10, 9 },
+  { 6, 12, 13, 8, 5, 9, 10, 7 },
+};
+HPRef_Struct reftet_1e_2vd =
+{
+  HP_TET,
+  reftet_1e_2vd_splitedges, 
+  0, 0,
+  reftet_1e_2vd_newelstypes, 
+  reftet_1e_2vd_newels
+};
+
+*/
+
+
+
+
+//  HP_TET_1E_2VD,  // 1 v on edge
+int reftet_1e_2vd_splitedges[][3] =
+{
+  // { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  // { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_1e_2vd_splitfaces[][4] =
+  {
+    { 1, 3, 4, 19 },
+    { 2, 3, 4, 22 },
+    { 3, 1, 4, 24 },
+    { 3, 2, 4, 25 },
+    { 4, 1, 3, 27 },
+    { 4, 2, 3, 28 },
+    { 0, 0, 0, 0 }
+  };
+HPREF_ELEMENT_TYPE reftet_1e_2vd_newelstypes[] =
+  {
+    HP_PRISM_SINGEDGE,
+    HP_TET_0E_1V,
+    HP_TET_0E_1V,
+    HP_PRISM,
+    HP_HEX,
+    HP_PYRAMID,
+    HP_HEX,
+    HP_PYRAMID,
+    HP_PRISM,
+    HP_PRISM,
+    HP_NONE,
+  };
+int reftet_1e_2vd_newels[][8] =
+{
+  { 1, 6, 7, 2, 9, 10 },
+  { 3, 11, 12, 13 },
+  { 4, 16, 15, 14 },
+  { 7, 6, 19, 10, 9, 22 },
+  { 7, 19, 27, 14, 10, 22, 28, 15 },
+  { 14, 15, 28, 27, 16 },
+  { 9, 6, 19, 22, 12, 11, 24, 25 },
+  { 12, 11, 24, 25, 13 },
+  { 19, 24, 27, 22, 25, 28 },
+  { 16, 28, 27, 13, 25, 24 }
+};
+HPRef_Struct reftet_1e_2vd =
+{
+  HP_TET,
+  reftet_1e_2vd_splitedges, 
+  reftet_1e_2vd_splitfaces, 
+  0,
+  reftet_1e_2vd_newelstypes, 
+  reftet_1e_2vd_newels
+};
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+// HP_TET_1E_3VA
+int reftet_1e_3va_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 0, 0, 0 }
+};
+int reftet_1e_3va_splitelements[][5] =
+{
+  { 1, 2, 3, 4, 14 },
+  { 0 }
+};
+
+HPREF_ELEMENT_TYPE reftet_1e_3va_newelstypes[] =
+{
+  HP_PRISM_SINGEDGE,
+  HP_TET_1E_1VA,
+  HP_TET_1E_1VA,
+  HP_TET_0E_1V,
+
+  HP_PYRAMID,
+  HP_TET,
+  HP_PYRAMID, 
+  HP_TET,
+  HP_PYRAMID,
+  HP_TET,
+  HP_PYRAMID,
+  HP_TET,
+  HP_NONE,
+};
+int reftet_1e_3va_newels[][8] =
+{
+  { 5, 6, 7, 8, 9, 10 },
+  { 1, 5, 6, 7 },
+  { 2, 8, 10, 9 },
+  { 3, 11, 12, 13 },
+
+  { 6, 7, 10, 9, 14 },
+  { 4, 10, 7, 14 },
+  { 9, 10, 13, 12, 14 },
+  { 4, 13, 10, 14 },
+  { 6, 11, 13, 7, 14 },
+  { 4, 7, 13, 14 },
+  { 6, 11, 12, 9, 14 },
+  { 11, 13, 12, 14 },
+};
+
+HPRef_Struct reftet_1e_3va =
+{
+  HP_TET,
+  reftet_1e_3va_splitedges, 
+  0,
+  reftet_1e_3va_splitelements, 
+  reftet_1e_3va_newelstypes, 
+  reftet_1e_3va_newels
+};
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+//  HP_TET_1E_3VB,  // 1 v on edge
+int reftet_1e_3vb_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  // { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_1e_3vb_splitfaces[][4] =
+  {
+    { 1, 3, 4, 19 },
+    { 2, 3, 4, 22 },
+    { 3, 1, 4, 24 },
+    { 3, 2, 4, 25 },
+    { 4, 1, 3, 27 },
+    { 4, 2, 3, 28 },
+    { 0, 0, 0, 0 }
+  };
+HPREF_ELEMENT_TYPE reftet_1e_3vb_newelstypes[] =
+  {
+    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_TET_0E_1V,
+    HP_TET_0E_1V,
+    HP_PRISM,
+    HP_HEX,
+    HP_PYRAMID,
+    HP_HEX,
+    HP_PYRAMID,
+    HP_PRISM,
+    HP_PRISM,
+    HP_NONE,
+  };
+int reftet_1e_3vb_newels[][8] =
+{
+  { 1, 5, 6, 7 },
+  { 5, 6, 7, 2, 9, 10 },
+  { 3, 11, 12, 13 },
+  { 4, 16, 15, 14 },
+  { 7, 6, 19, 10, 9, 22 },
+  { 7, 19, 27, 14, 10, 22, 28, 15 },
+  { 14, 15, 28, 27, 16 },
+  { 9, 6, 19, 22, 12, 11, 24, 25 },
+  { 12, 11, 24, 25, 13 },
+  { 19, 24, 27, 22, 25, 28 },
+  { 16, 28, 27, 13, 25, 24 }
+};
+HPRef_Struct reftet_1e_3vb =
+{
+  HP_TET,
+  reftet_1e_3vb_splitedges, 
+  reftet_1e_3vb_splitfaces, 
+  0,
+  reftet_1e_3vb_newelstypes, 
+  reftet_1e_3vb_newels
+};
+
+
+
+
+
+
+
+
+
+// HP_TET_1E_4V
+int reftet_1e_4v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_1e_4v_splitfaces[][4] =
+  {
+    { 1, 2, 3, 17 },
+    { 1, 2, 4, 18 },
+    { 1, 3, 4, 19 },
+
+    { 2, 1, 3, 20 },
+    { 2, 1, 4, 21 },
+    { 2, 3, 4, 22 },
+
+    { 3, 1, 2, 23 },
+    { 3, 1, 4, 24 },
+    { 3, 2, 4, 25 },
+
+    { 4, 1, 2, 26 },
+    { 4, 1, 3, 27 },
+    { 4, 2, 3, 28 },
+    { 0, 0, 0, 0 },
+  };
+int reftet_1e_4v_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 29 },
+    { 2, 3, 4, 1, 30 },
+    { 3, 4, 1, 2, 31 },
+    { 4, 1, 2, 3, 32 },
+    { 0 },
+  };
+HPREF_ELEMENT_TYPE reftet_1e_4v_newelstypes[] =
+{
+  HP_HEX_1E_1V,
+  HP_HEX_1E_1V,
+  HP_HEX_0E_1V,
+  HP_HEX_0E_1V,
+  HP_PRISM_SINGEDGE, HP_PRISM, 
+  HP_PRISM, HP_PRISM, 
+  HP_PRISM, HP_PRISM, 
+  HP_PRISM, HP_PRISM, 
+  HP_PRISM, HP_PRISM, 
+  HP_PRISM, HP_PRISM, 
+  HP_PRISM,
+  HP_PRISM,
+  HP_PRISM,
+  HP_PRISM,
+  HP_TET,
+  HP_NONE,
+};
+int reftet_1e_4v_newels[][8] =
+{
+  { 1, 5, 17, 6, 7, 18, 29, 19 },
+  //  { 2, 9, 20, 8, 10, 22, 30, 21 },
+  { 2, 8, 21, 10, 9, 20, 30, 22 },
+  { 3, 11, 23, 12, 13, 24, 31, 25 },
+  { 4, 15, 26, 14, 16, 28, 32, 27 },
+  { 5, 17, 18, 8, 20, 21 },
+  { 18, 17, 29, 21, 20, 30 },
+  { 6, 19, 17,  11, 24, 23 },
+  { 17, 19, 29,  23, 24, 31 },
+  { 7, 18, 19, 14, 26, 27 },
+  { 19, 18, 29, 27, 26, 32 },
+  { 9, 20, 22, 12, 23, 25 },
+  { 22, 20, 30, 25, 23, 31 },
+  { 10, 22, 21, 15, 28, 26 },
+  { 21, 22, 30, 26, 28, 32 },
+  { 13, 24, 25, 16, 27, 28 },
+  { 25, 24, 31, 28, 27, 32 },
+  /*
+  { 5, 17, 29, 18, 8, 20, 30, 21 },
+  { 6, 19, 29, 17, 11, 24, 31, 23 },
+  { 7, 18, 29, 19, 14, 26, 32, 27 },
+  { 9, 20, 30, 22, 12, 23, 31, 25 },
+  { 10, 22, 30, 21, 15, 28, 32, 26 },
+  { 13, 24, 31, 25, 16, 27, 32, 28 },
+  */
+  { 17, 20, 23, 29, 30, 31 },
+  { 18, 26, 21, 29, 32, 30 },
+  { 19, 24, 27, 29, 31, 32 },
+  { 22, 28, 25, 30, 32, 31 },
+
+  { 29, 30, 31, 32 },
+};
+HPRef_Struct reftet_1e_4v =
+{
+  HP_TET,
+  reftet_1e_4v_splitedges, 
+  reftet_1e_4v_splitfaces, 
+  reftet_1e_4v_splitelements, 
+  reftet_1e_4v_newelstypes, 
+  reftet_1e_4v_newels
+};
+
+
+
+
+
+
+
+
+
+
+
+
+//  HP_TET_2EA_0V,  // 2 edges connected
+int reftet_2ea_0v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 0, 0, 0 }
+};
+int reftet_2ea_0v_splitfaces[][4] =
+  {
+    { 1, 2, 3, 17 },
+    { 0, 0, 0, 0 }
+  };
+HPREF_ELEMENT_TYPE reftet_2ea_0v_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM,
+    HP_TET,
+    HP_NONE,
+  };
+int reftet_2ea_0v_newels[][8] =
+{
+  { 1, 5, 17, 6, 7 },
+  { 5, 17, 7, 2, 9, 10 },
+  { 6, 7, 17, 3, 13, 12 },
+  { 17, 9, 12, 7, 10, 13 },
+  { 7, 10, 13, 4 },
+};
+HPRef_Struct reftet_2ea_0v =
+{
+  HP_TET,
+  reftet_2ea_0v_splitedges, 
+  reftet_2ea_0v_splitfaces, 
+  0,
+  reftet_2ea_0v_newelstypes, 
+  reftet_2ea_0v_newels
+};
+
+
+
+
+
+
+//  HP_TET_2EA_1VA,  // 2 edges connected
+int reftet_2ea_1va_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 0, 0, 0 }
+};
+int reftet_2ea_1va_splitfaces[][4] =
+  {
+    { 1, 2, 3, 17 },
+    { 0, 0, 0, 0 }
+  };
+HPREF_ELEMENT_TYPE reftet_2ea_1va_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    HP_PRISM_SINGEDGE,
+    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM,
+    HP_TET,
+    HP_NONE,
+  };
+int reftet_2ea_1va_newels[][8] =
+{
+  { 1, 5, 17, 6, 7 },
+  { 5, 17, 7, 8, 9, 10 },
+  { 2, 8, 10, 9 },
+  { 6, 7, 17, 3, 13, 12 },
+  { 17, 9, 12, 7, 10, 13 },
+  { 7, 10, 13, 4 },
+};
+HPRef_Struct reftet_2ea_1va =
+{
+  HP_TET,
+  reftet_2ea_1va_splitedges, 
+  reftet_2ea_1va_splitfaces, 
+  0,
+  reftet_2ea_1va_newelstypes, 
+  reftet_2ea_1va_newels
+};
+
+
+
+
+
+
+
+
+//  HP_TET_2EA_1VB, 
+int reftet_2ea_1vb_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 0, 0, 0 }
+};
+int reftet_2ea_1vb_splitfaces[][4] =
+  {
+    { 1, 2, 3, 17 },
+    { 0, 0, 0, 0 }
+  };
+HPREF_ELEMENT_TYPE reftet_2ea_1vb_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM,
+    HP_TET,
+    HP_NONE,
+  };
+int reftet_2ea_1vb_newels[][8] =
+{
+  { 1, 5, 17, 6, 7 },
+  { 3, 11, 12, 13 },
+  { 5, 17, 7, 2, 9, 10 },
+  { 6, 7, 17, 11, 13, 12 },
+  { 17, 9, 12, 7, 10, 13 },
+  { 7, 10, 13, 4 },
+};
+HPRef_Struct reftet_2ea_1vb =
+{
+  HP_TET,
+  reftet_2ea_1vb_splitedges, 
+  reftet_2ea_1vb_splitfaces, 
+  0,
+  reftet_2ea_1vb_newelstypes, 
+  reftet_2ea_1vb_newels
+};
+
+
+
+
+
+
+//  HP_TET_2EA_1VC,  // 2 edges connected
+int reftet_2ea_1vc_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  //  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_2ea_1vc_splitfaces[][4] =
+  {
+    { 1, 2, 3, 17 },
+    { 2, 3, 4, 18 },
+    { 3, 4, 2, 19 },
+    { 4, 2, 3, 20 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_2ea_1vc_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 21 },
+    { 0, 0, 0, 0 }
+  };
+HPREF_ELEMENT_TYPE reftet_2ea_1vc_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    //    HP_TET_1E_1VA,
+    HP_TET_0E_1V,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+
+    HP_TET, HP_TET, HP_TET, HP_TET, 
+    HP_PYRAMID, HP_PYRAMID, HP_PYRAMID, 
+    HP_PYRAMID, HP_PYRAMID, HP_TET,
+    HP_PYRAMID, HP_PYRAMID, HP_TET,
+    //     HP_PRISM,
+    //    HP_PRISM,
+    HP_NONE,
+  };
+int reftet_2ea_1vc_newels[][8] =
+{
+  { 1, 5, 17, 6, 7 },
+  // { 3, 11, 12, 13 },
+  { 4, 15, 14, 16 }, 
+  { 5, 17, 7, 2, 9, 10 },
+  { 6, 7, 17, 3, 13, 12 },
+ 
+  { 9, 10, 18, 21 },
+  { 13, 12, 19, 21 },
+  { 15, 16, 20, 21 },
+  { 18, 20, 19, 21 },
+  { 10, 15, 20, 18, 21 },
+  { 13, 19, 20, 16, 21 },
+  { 9, 18, 19, 12, 21 },
+  
+  { 7, 13, 16, 14, 21 },
+  { 7, 14, 15, 10, 21 },
+  { 9, 12, 17, 21 },
+  { 7, 10, 9, 17, 21 },
+  { 7, 17, 12, 13, 21 },
+  { 14, 16, 15, 21 },
+  //  { 17, 9, 12, 7, 10, 13 },
+  //  { 7, 10, 13, 14, 15, 16 },
+};
+HPRef_Struct reftet_2ea_1vc =
+{
+  HP_TET,
+  reftet_2ea_1vc_splitedges, 
+  reftet_2ea_1vc_splitfaces, 
+  reftet_2ea_1vc_splitelements, 
+  reftet_2ea_1vc_newelstypes, 
+  reftet_2ea_1vc_newels
+};
+
+
+
+
+
+
+
+
+
+
+
+ 
+//  HP_TET_2EA_2VA, 
+int reftet_2ea_2va_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 0, 0, 0 }
+};
+int reftet_2ea_2va_splitfaces[][4] =
+  {
+    { 1, 2, 3, 17 },
+    { 0, 0, 0, 0 }
+  };
+HPREF_ELEMENT_TYPE reftet_2ea_2va_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    HP_TET_1E_1VA,
+    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM,
+    HP_TET,
+    HP_NONE,
+  };
+int reftet_2ea_2va_newels[][8] =
+{
+  { 1, 5, 17, 6, 7 },
+  { 3, 11, 12, 13 },
+  { 2, 8, 10, 9 },
+  { 5, 17, 7, 8, 9, 10 },
+  { 6, 7, 17, 11, 13, 12 },
+  { 17, 9, 12, 7, 10, 13 },
+  { 7, 10, 13, 4 },
+};
+HPRef_Struct reftet_2ea_2va =
+{
+  HP_TET,
+  reftet_2ea_2va_splitedges, 
+  reftet_2ea_2va_splitfaces, 
+  0,
+  reftet_2ea_2va_newelstypes, 
+  reftet_2ea_2va_newels
+};
+
+
+
+
+
+
+
+
+
+
+
+//  HP_TET_2EA_2VB,  // 2 edges connected
+int reftet_2ea_2vb_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  //  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_2ea_2vb_splitfaces[][4] =
+  {
+    { 1, 2, 3, 17 },
+    { 2, 3, 4, 18 },
+    { 3, 4, 2, 19 },
+    { 4, 2, 3, 20 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_2ea_2vb_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 21 },
+    { 0, 0, 0, 0 }
+  };
+HPREF_ELEMENT_TYPE reftet_2ea_2vb_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    HP_TET_1E_1VA,
+    //  HP_TET_1E_1VA,
+    HP_TET_0E_1V,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+
+    HP_TET, HP_TET, HP_TET, HP_TET, 
+    HP_PYRAMID, HP_PYRAMID, HP_PYRAMID, 
+    HP_PYRAMID, HP_PYRAMID, HP_TET,
+    HP_PYRAMID, HP_PYRAMID, HP_TET,
+    //     HP_PRISM,
+    //    HP_PRISM,
+    HP_NONE,
+  };
+int reftet_2ea_2vb_newels[][8] =
+{
+  { 1, 5, 17, 6, 7 },
+  { 2, 8, 10, 9 },
+  //  { 3, 11, 12, 13 },
+  { 4, 15, 14, 16 }, 
+  { 5, 17, 7, 8, 9, 10 },
+  { 6, 7, 17, 3, 13, 12 },
+ 
+  { 9, 10, 18, 21 },
+  { 13, 12, 19, 21 },
+  { 15, 16, 20, 21 },
+  { 18, 20, 19, 21 },
+  { 10, 15, 20, 18, 21 },
+  { 13, 19, 20, 16, 21 },
+  { 9, 18, 19, 12, 21 },
+  
+  { 7, 13, 16, 14, 21 },
+  { 7, 14, 15, 10, 21 },
+  { 9, 12, 17, 21 },
+  { 7, 10, 9, 17, 21 },
+  { 7, 17, 12, 13, 21 },
+  { 14, 16, 15, 21 },
+  //  { 17, 9, 12, 7, 10, 13 },
+  //  { 7, 10, 13, 14, 15, 16 },
+};
+HPRef_Struct reftet_2ea_2vb =
+{
+  HP_TET,
+  reftet_2ea_2vb_splitedges, 
+  reftet_2ea_2vb_splitfaces, 
+  reftet_2ea_2vb_splitelements, 
+  reftet_2ea_2vb_newelstypes, 
+  reftet_2ea_2vb_newels
+};
+
+
+
+
+
+
+
+ 
+
+
+//  HP_TET_2EA_2VC,  // 2 edges connected
+int reftet_2ea_2vc_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  //  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_2ea_2vc_splitfaces[][4] =
+  {
+    { 1, 2, 3, 17 },
+    { 2, 3, 4, 18 },
+    { 3, 4, 2, 19 },
+    { 4, 2, 3, 20 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_2ea_2vc_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 21 },
+    { 0, 0, 0, 0 }
+  };
+HPREF_ELEMENT_TYPE reftet_2ea_2vc_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    HP_TET_1E_1VA,
+    //    HP_TET_1E_1VA,
+    HP_TET_0E_1V,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+
+    HP_TET, HP_TET, HP_TET, HP_TET, 
+    HP_PYRAMID, HP_PYRAMID, HP_PYRAMID, 
+    HP_PYRAMID, HP_PYRAMID, HP_TET,
+    HP_PYRAMID, HP_PYRAMID, HP_TET,
+    //     HP_PRISM,
+    //    HP_PRISM,
+    HP_NONE,
+  };
+int reftet_2ea_2vc_newels[][8] =
+{
+  { 1, 5, 17, 6, 7 },
+  //  { 2, 8, 10, 9 },
+  { 3, 11, 12, 13 },
+  { 4, 15, 14, 16 }, 
+  { 5, 17, 7, 2, 9, 10 },
+  { 6, 7, 17, 11, 13, 12 },
+ 
+  { 9, 10, 18, 21 },
+  { 13, 12, 19, 21 },
+  { 15, 16, 20, 21 },
+  { 18, 20, 19, 21 },
+  { 10, 15, 20, 18, 21 },
+  { 13, 19, 20, 16, 21 },
+  { 9, 18, 19, 12, 21 },
+  
+  { 7, 13, 16, 14, 21 },
+  { 7, 14, 15, 10, 21 },
+  { 9, 12, 17, 21 },
+  { 7, 10, 9, 17, 21 },
+  { 7, 17, 12, 13, 21 },
+  { 14, 16, 15, 21 },
+  //  { 17, 9, 12, 7, 10, 13 },
+  //  { 7, 10, 13, 14, 15, 16 },
+};
+HPRef_Struct reftet_2ea_2vc =
+{
+  HP_TET,
+  reftet_2ea_2vc_splitedges, 
+  reftet_2ea_2vc_splitfaces, 
+  reftet_2ea_2vc_splitelements, 
+  reftet_2ea_2vc_newelstypes, 
+  reftet_2ea_2vc_newels
+};
+
+
+
+
+
+
+
+
+//  HP_TET_2EA_3V,  // 2 edges connected
+int reftet_2ea_3v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_2ea_3v_splitfaces[][4] =
+  {
+    { 1, 2, 3, 17 },
+    { 2, 3, 4, 18 },
+    { 3, 4, 2, 19 },
+    { 4, 2, 3, 20 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_2ea_3v_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 21 },
+    { 0, 0, 0, 0 }
+  };
+HPREF_ELEMENT_TYPE reftet_2ea_3v_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    HP_TET_1E_1VA,
+    HP_TET_1E_1VA,
+    HP_TET_0E_1V,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+
+    HP_TET, HP_TET, HP_TET, HP_TET, 
+    HP_PYRAMID, HP_PYRAMID, HP_PYRAMID, 
+    HP_PYRAMID, HP_PYRAMID, HP_TET,
+    HP_PYRAMID, HP_PYRAMID, HP_TET,
+    //     HP_PRISM,
+    //    HP_PRISM,
+    HP_NONE,
+  };
+int reftet_2ea_3v_newels[][8] =
+{
+  { 1, 5, 17, 6, 7 },
+  { 2, 8, 10, 9 },
+  { 3, 11, 12, 13 },
+  { 4, 15, 14, 16 }, 
+  { 5, 17, 7, 8, 9, 10 },
+  { 6, 7, 17, 11, 13, 12 },
+ 
+  { 9, 10, 18, 21 },
+  { 13, 12, 19, 21 },
+  { 15, 16, 20, 21 },
+  { 18, 20, 19, 21 },
+  { 10, 15, 20, 18, 21 },
+  { 13, 19, 20, 16, 21 },
+  { 9, 18, 19, 12, 21 },
+  
+  { 7, 13, 16, 14, 21 },
+  { 7, 14, 15, 10, 21 },
+  { 9, 12, 17, 21 },
+  { 7, 10, 9, 17, 21 },
+  { 7, 17, 12, 13, 21 },
+  { 14, 16, 15, 21 },
+  //  { 17, 9, 12, 7, 10, 13 },
+  //  { 7, 10, 13, 14, 15, 16 },
+};
+HPRef_Struct reftet_2ea_3v =
+{
+  HP_TET,
+  reftet_2ea_3v_splitedges, 
+  reftet_2ea_3v_splitfaces, 
+  reftet_2ea_3v_splitelements, 
+  reftet_2ea_3v_newelstypes, 
+  reftet_2ea_3v_newels
+};
+
+
+
+
+
+
+
+//  HP_TET_2EB_0V,  // 2 opposite edges
+int reftet_2eb_0v_splitedges[][3] =
+{
+  { 1, 3, 5 },
+  { 1, 4, 6 },
+  { 2, 3, 7 },
+  { 2, 4, 8 },
+  { 3, 1, 9 },
+  { 3, 2, 10 },
+  { 4, 1, 11 },
+  { 4, 2, 12 },
+  { 0, 0, 0 }
+};
+
+HPREF_ELEMENT_TYPE reftet_2eb_0v_newelstypes[] =
+  {
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    HP_HEX,
+    HP_NONE,
+  };
+int reftet_2eb_0v_newels[][8] =
+{
+  { 1, 5, 6, 2, 7, 8 },
+  { 3, 9, 10, 4, 11, 12 },
+  { 6, 11, 12, 8, 5, 9, 10, 7 },
+};
+HPRef_Struct reftet_2eb_0v =
+{
+  HP_TET,
+  reftet_2eb_0v_splitedges, 
+  0, 0,
+  reftet_2eb_0v_newelstypes, 
+  reftet_2eb_0v_newels
+};
+
+
+
+
+
+
+
+//  HP_TET_2EB_2VA,  // 2 opposite edges
+int reftet_2eb_2va_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+
+HPREF_ELEMENT_TYPE reftet_2eb_2va_newelstypes[] =
+  {
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    HP_TET_1E_1VA,
+    //    HP_TET_1E_1VA,
+    HP_TET_1E_1VA,
+    //    HP_TET_1E_1VA,
+    HP_HEX,
+    HP_NONE,
+  };
+int reftet_2eb_2va_newels[][8] =
+{
+  { 5, 6, 7, 2, 9, 10 },
+  { 4, 15, 14, 13, 12, 11 },
+  { 1, 5, 6, 7 },
+  //  { 2, 8, 10, 9 },
+  { 3, 13, 11, 12 },
+  //  { 4, 16, 15, 14 },
+  { 7, 14, 15, 10, 6, 11, 12, 9 }
+};
+HPRef_Struct reftet_2eb_2va =
+{
+  HP_TET,
+  reftet_2eb_2va_splitedges, 
+  0, 0,
+  reftet_2eb_2va_newelstypes, 
+  reftet_2eb_2va_newels
+};
+
+
+
+
+
+
+//  HP_TET_2EB_4V,  // 2 opposite edges
+int reftet_2eb_4v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+
+HPREF_ELEMENT_TYPE reftet_2eb_4v_newelstypes[] =
+  {
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    HP_TET_1E_1VA,
+    HP_TET_1E_1VA,
+    HP_TET_1E_1VA,
+    HP_TET_1E_1VA,
+    HP_HEX,
+    HP_NONE,
+  };
+int reftet_2eb_4v_newels[][8] =
+{
+  { 5, 6, 7, 8, 9, 10 },
+  { 16, 15, 14, 13, 12, 11 },
+  { 1, 5, 6, 7 },
+  { 2, 8, 10, 9 },
+  { 3, 13, 11, 12 },
+  { 4, 16, 15, 14 },
+  { 7, 14, 15, 10, 6, 11, 12, 9 }
+};
+HPRef_Struct reftet_2eb_4v =
+{
+  HP_TET,
+  reftet_2eb_4v_splitedges, 
+  0, 0,
+  reftet_2eb_4v_newelstypes, 
+  reftet_2eb_4v_newels
+};
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+//  HP_TET_3EA_0V,  
+int reftet_3ea_0v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 3, 8 },
+  { 2, 4, 9 },
+  { 3, 2, 10 },
+  { 3, 4, 11 },
+  { 4, 2, 12 },
+  { 4, 3, 13 },
+  { 0, 0, 0 }
+};
+int reftet_3ea_0v_splitfaces[][4] =
+  {
+    { 1, 2, 3, 14 },
+    { 1, 2, 4, 15 },
+    { 1, 3, 4, 16 },
+    { 2, 3, 4, 17 },
+    { 3, 4, 2, 18 },
+    { 4, 2, 3, 19 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_3ea_0v_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 20 },
+    { 0 },
+  };
+
+HPREF_ELEMENT_TYPE reftet_3ea_0v_newelstypes[] =
+  {
+    HP_HEX_3E_0V,
+    HP_HEX_1E_0V,
+    HP_HEX_1E_0V,
+    HP_HEX_1E_0V,
+    HP_PRISM,
+    HP_PRISM,
+    HP_PRISM,
+    HP_TET,
+    HP_NONE,
+  };
+int reftet_3ea_0v_newels[][8] =
+{
+  { 1, 5, 14, 6, 7, 15, 20, 16 },
+  { 5, 2, 8, 14, 15, 9, 17, 20 },
+  { 3, 6, 14, 10, 11, 16, 20, 18 },
+  { 7, 4, 12, 15, 16, 13, 19, 20 },
+  { 11, 13, 16, 18, 19, 20 },
+  { 15, 12, 9, 20, 19, 17 },
+  { 8, 10, 14, 17, 18, 20 },
+  { 20, 17, 18, 19 },
+};
+HPRef_Struct reftet_3ea_0v =
+{
+  HP_TET,
+  reftet_3ea_0v_splitedges, 
+  reftet_3ea_0v_splitfaces, 
+  reftet_3ea_0v_splitelements, 
+  reftet_3ea_0v_newelstypes, 
+  reftet_3ea_0v_newels
+};
+
+
+
+
+
+
+
+
+
+
+//  HP_TET_3EA_1V,  
+int reftet_3ea_1v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 3, 8 },
+  { 2, 4, 9 },
+  { 3, 2, 10 },
+  { 3, 4, 11 },
+  { 4, 2, 12 },
+  { 4, 3, 13 },
+  { 2, 1, 21 },
+  { 3, 1, 22 },
+  { 4, 1, 23 },
+  { 0, 0, 0 }
+};
+int reftet_3ea_1v_splitfaces[][4] =
+  {
+    { 1, 2, 3, 14 },
+    { 1, 2, 4, 15 },
+    { 1, 3, 4, 16 },
+    { 2, 3, 4, 17 },
+    { 3, 4, 2, 18 },
+    { 4, 2, 3, 19 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_3ea_1v_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 20 },
+    { 0 },
+  };
+
+HPREF_ELEMENT_TYPE reftet_3ea_1v_newelstypes[] =
+  {
+    HP_HEX_3E_0V,
+    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM,
+    //    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM,
+    //    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM,
+
+    HP_PRISM,
+    HP_PRISM,
+    HP_PRISM,
+    HP_TET,
+    HP_NONE,
+  };
+int reftet_3ea_1v_newels[][8] =
+{
+  { 1, 5, 14, 6, 7, 15, 20, 16 },
+
+  { 2, 21, 9, 8 },
+  { 5, 14, 15, 21, 8, 9 },
+  { 15, 14, 20, 9, 8, 17 },
+  //  { 3, 22, 10, 11 },
+  //  { 6, 16, 14, 22, 11, 10 },
+  { 6, 16, 14, 3, 11, 10 },
+  { 14, 16, 20, 10, 11, 18 },
+  //  { 4, 23, 13, 12 },
+  //  { 7, 15, 16, 23, 12, 13 },
+  { 7, 15, 16, 4, 12, 13 },
+  { 16, 15, 20, 13, 12, 19 },
+
+  { 11, 13, 16, 18, 19, 20 },
+  { 15, 12, 9, 20, 19, 17 },
+  { 8, 10, 14, 17, 18, 20 },
+  { 20, 17, 18, 19 },
+};
+HPRef_Struct reftet_3ea_1v =
+{
+  HP_TET,
+  reftet_3ea_1v_splitedges, 
+  reftet_3ea_1v_splitfaces, 
+  reftet_3ea_1v_splitelements, 
+  reftet_3ea_1v_newelstypes, 
+  reftet_3ea_1v_newels
+};
+
+
+
+
+
+
+
+
+
+
+//  HP_TET_3EA_2V,  
+int reftet_3ea_2v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 3, 8 },
+  { 2, 4, 9 },
+  { 3, 2, 10 },
+  { 3, 4, 11 },
+  { 4, 2, 12 },
+  { 4, 3, 13 },
+  { 2, 1, 21 },
+  { 3, 1, 22 },
+  { 4, 1, 23 },
+  { 0, 0, 0 }
+};
+int reftet_3ea_2v_splitfaces[][4] =
+  {
+    { 1, 2, 3, 14 },
+    { 1, 2, 4, 15 },
+    { 1, 3, 4, 16 },
+    { 2, 3, 4, 17 },
+    { 3, 4, 2, 18 },
+    { 4, 2, 3, 19 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_3ea_2v_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 20 },
+    { 0 },
+  };
+
+HPREF_ELEMENT_TYPE reftet_3ea_2v_newelstypes[] =
+  {
+    HP_HEX_3E_0V,
+    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM,
+    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM,
+    //    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM,
+
+    HP_PRISM,
+    HP_PRISM,
+    HP_PRISM,
+    HP_TET,
+    HP_NONE,
+  };
+int reftet_3ea_2v_newels[][8] =
+{
+  { 1, 5, 14, 6, 7, 15, 20, 16 },
+
+  { 2, 21, 9, 8 },
+  { 5, 14, 15, 21, 8, 9 },
+  { 15, 14, 20, 9, 8, 17 },
+  { 3, 22, 10, 11 },
+  { 6, 16, 14, 22, 11, 10 },
+  { 14, 16, 20, 10, 11, 18 },
+  //  { 4, 23, 13, 12 },
+  { 7, 15, 16, 4, 12, 13 },
+  { 16, 15, 20, 13, 12, 19 },
+
+  { 11, 13, 16, 18, 19, 20 },
+  { 15, 12, 9, 20, 19, 17 },
+  { 8, 10, 14, 17, 18, 20 },
+  { 20, 17, 18, 19 },
+};
+HPRef_Struct reftet_3ea_2v =
+{
+  HP_TET,
+  reftet_3ea_2v_splitedges, 
+  reftet_3ea_2v_splitfaces, 
+  reftet_3ea_2v_splitelements, 
+  reftet_3ea_2v_newelstypes, 
+  reftet_3ea_2v_newels
+};
+
+
+
+
+
+
+
+
+//  HP_TET_3EA_3V,  
+int reftet_3ea_3v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 3, 8 },
+  { 2, 4, 9 },
+  { 3, 2, 10 },
+  { 3, 4, 11 },
+  { 4, 2, 12 },
+  { 4, 3, 13 },
+  { 2, 1, 21 },
+  { 3, 1, 22 },
+  { 4, 1, 23 },
+  { 0, 0, 0 }
+};
+int reftet_3ea_3v_splitfaces[][4] =
+  {
+    { 1, 2, 3, 14 },
+    { 1, 2, 4, 15 },
+    { 1, 3, 4, 16 },
+    { 2, 3, 4, 17 },
+    { 3, 4, 2, 18 },
+    { 4, 2, 3, 19 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_3ea_3v_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 20 },
+    { 0 },
+  };
+
+HPREF_ELEMENT_TYPE reftet_3ea_3v_newelstypes[] =
+  {
+    HP_HEX_3E_0V,
+    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM,
+    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM,
+    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM,
+
+    HP_PRISM,
+    HP_PRISM,
+    HP_PRISM,
+    HP_TET,
+    HP_NONE,
+  };
+int reftet_3ea_3v_newels[][8] =
+{
+  { 1, 5, 14, 6, 7, 15, 20, 16 },
+
+  { 2, 21, 9, 8 },
+  { 5, 14, 15, 21, 8, 9 },
+  { 15, 14, 20, 9, 8, 17 },
+  { 3, 22, 10, 11 },
+  { 6, 16, 14, 22, 11, 10 },
+  { 14, 16, 20, 10, 11, 18 },
+  { 4, 23, 13, 12 },
+  { 7, 15, 16, 23, 12, 13 },
+  { 16, 15, 20, 13, 12, 19 },
+
+  { 11, 13, 16, 18, 19, 20 },
+  { 15, 12, 9, 20, 19, 17 },
+  { 8, 10, 14, 17, 18, 20 },
+  { 20, 17, 18, 19 },
+};
+HPRef_Struct reftet_3ea_3v =
+{
+  HP_TET,
+  reftet_3ea_3v_splitedges, 
+  reftet_3ea_3v_splitfaces, 
+  reftet_3ea_3v_splitelements, 
+  reftet_3ea_3v_newelstypes, 
+  reftet_3ea_3v_newels
+};
+
+
+
+
+
+
+
+//  HP_TET_3EV_0V,  
+int reftet_3eb_0v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  //  { 3, 2, 12 },
+  { 3, 4, 13 },
+  //  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_3eb_0v_splitfaces[][4] =
+  {
+    { 1, 2, 4, 17 },
+    { 2, 1, 3, 18 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_3eb_0v_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 20 },
+    { 0 },
+  };
+
+HPREF_ELEMENT_TYPE reftet_3eb_0v_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    HP_PYRAMID_EDGES,
+    //    HP_TET_1E_1VA,
+    //    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_TET,
+    HP_TET,
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_NONE,
+  };
+int reftet_3eb_0v_newels[][8] =
+{
+  { 1, 7, 17, 5, 6 },
+  { 2, 9, 18, 8, 10 },
+  //  { 3, 12, 13, 11 },
+  //  { 4, 14, 16, 15 },
+  { 5, 6, 17, 8, 18, 10 },
+  { 7, 17, 6, 4, 15, 16 },
+  { 9, 18, 10, 3, 11, 13 },
+  
+  { 10, 15, 16, 13, 20 },
+  { 6, 11, 13, 16, 20 },
+  { 10, 17, 15, 20 },
+  { 6, 18, 11, 20 },
+  { 6, 17, 10, 18, 20 },
+  { 6, 16, 15, 17, 20 },
+  { 18, 10, 13, 11, 20 },
+};
+HPRef_Struct reftet_3eb_0v =
+{
+  HP_TET,
+  reftet_3eb_0v_splitedges, 
+  reftet_3eb_0v_splitfaces, 
+  reftet_3eb_0v_splitelements, 
+  reftet_3eb_0v_newelstypes, 
+  reftet_3eb_0v_newels
+};
+
+
+
+
+
+
+
+
+
+//  HP_TET_3EV_1V,  
+int reftet_3eb_1v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  //  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_3eb_1v_splitfaces[][4] =
+  {
+    { 1, 2, 4, 17 },
+    { 2, 1, 3, 18 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_3eb_1v_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 20 },
+    { 0 },
+  };
+
+HPREF_ELEMENT_TYPE reftet_3eb_1v_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    HP_PYRAMID_EDGES,
+    HP_TET_1E_1VA,
+    //    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_TET,
+    HP_TET,
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_NONE,
+  };
+int reftet_3eb_1v_newels[][8] =
+{
+  { 1, 7, 17, 5, 6 },
+  { 2, 9, 18, 8, 10 },
+  { 3, 12, 13, 11 },
+  //  { 4, 14, 16, 15 },
+  { 5, 6, 17, 8, 18, 10 },
+  { 7, 17, 6, 4, 15, 16 },
+  { 9, 18, 10, 12, 11, 13 },
+  
+  { 10, 15, 16, 13, 20 },
+  { 6, 11, 13, 16, 20 },
+  { 10, 17, 15, 20 },
+  { 6, 18, 11, 20 },
+  { 6, 17, 10, 18, 20 },
+  { 6, 16, 15, 17, 20 },
+  { 18, 10, 13, 11, 20 },
+};
+HPRef_Struct reftet_3eb_1v =
+{
+  HP_TET,
+  reftet_3eb_1v_splitedges, 
+  reftet_3eb_1v_splitfaces, 
+  reftet_3eb_1v_splitelements, 
+  reftet_3eb_1v_newelstypes, 
+  reftet_3eb_1v_newels
+};
+
+
+
+
+
+
+
+
+//  HP_TET_3EV_2V,  
+int reftet_3eb_2v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_3eb_2v_splitfaces[][4] =
+  {
+    { 1, 2, 4, 17 },
+    { 2, 1, 3, 18 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_3eb_2v_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 20 },
+    { 0 },
+  };
+
+HPREF_ELEMENT_TYPE reftet_3eb_2v_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    HP_PYRAMID_EDGES,
+    HP_TET_1E_1VA,
+    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_TET,
+    HP_TET,
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_NONE,
+  };
+int reftet_3eb_2v_newels[][8] =
+{
+  { 1, 7, 17, 5, 6 },
+  { 2, 9, 18, 8, 10 },
+  { 3, 12, 13, 11 },
+  { 4, 14, 16, 15 },
+  { 5, 6, 17, 8, 18, 10 },
+  { 7, 17, 6, 14, 15, 16 },
+  { 9, 18, 10, 12, 11, 13 },
+  
+  { 10, 15, 16, 13, 20 },
+  { 6, 11, 13, 16, 20 },
+  { 10, 17, 15, 20 },
+  { 6, 18, 11, 20 },
+  { 6, 17, 10, 18, 20 },
+  { 6, 16, 15, 17, 20 },
+  { 18, 10, 13, 11, 20 },
+};
+HPRef_Struct reftet_3eb_2v =
+{
+  HP_TET,
+  reftet_3eb_2v_splitedges, 
+  reftet_3eb_2v_splitfaces, 
+  reftet_3eb_2v_splitelements, 
+  reftet_3eb_2v_newelstypes, 
+  reftet_3eb_2v_newels
+};
+
+
+
+
+
+
+
+
+
+
+
+
+
+//  HP_TET_3EC_0V,  
+int reftet_3ec_0v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  //  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  //  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_3ec_0v_splitfaces[][4] =
+  {
+    { 1, 2, 3, 17 },
+    { 2, 1, 4, 18 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_3ec_0v_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 20 },
+    { 0 },
+  };
+
+HPREF_ELEMENT_TYPE reftet_3ec_0v_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    HP_PYRAMID_EDGES,
+    //    HP_TET_1E_1VA,
+    //    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_TET,
+    HP_TET,
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_NONE,
+  };
+int reftet_3ec_0v_newels[][8] =
+{
+  { 1, 5, 17, 6, 7 },
+  { 2, 8, 18, 10, 9 },
+  //  { 3, 11, 12, 13 },
+  //  { 4, 15, 14, 16 },
+  { 5, 17, 7, 8, 9, 18 },
+  { 6, 7, 17, 3, 13, 12 },
+  { 10, 9, 18, 4, 16, 14 },
+  
+  { 9, 16, 13, 12, 20 },
+  { 7, 13, 16, 14, 20 },
+  { 7, 14, 18, 20 },
+  { 9, 12, 17, 20 },
+  { 17, 7, 18, 9, 20 },
+  { 7, 17, 12, 13, 20 },
+  { 9, 18, 14, 16, 20 },
+};
+HPRef_Struct reftet_3ec_0v =
+{
+  HP_TET,
+  reftet_3ec_0v_splitedges, 
+  reftet_3ec_0v_splitfaces, 
+  reftet_3ec_0v_splitelements, 
+  reftet_3ec_0v_newelstypes, 
+  reftet_3ec_0v_newels
+};
+
+
+
+
+
+
+ 
+
+
+//  HP_TET_3EC_1V,  
+int reftet_3ec_1v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  // { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_3ec_1v_splitfaces[][4] =
+  {
+    { 1, 2, 3, 17 },
+    { 2, 1, 4, 18 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_3ec_1v_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 20 },
+    { 0 },
+  };
+
+HPREF_ELEMENT_TYPE reftet_3ec_1v_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    HP_PYRAMID_EDGES,
+    HP_TET_1E_1VA,
+    //    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_TET,
+    HP_TET,
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_NONE,
+  };
+int reftet_3ec_1v_newels[][8] =
+{
+  { 1, 5, 17, 6, 7 },
+  { 2, 8, 18, 10, 9 },
+  { 3, 11, 12, 13 },
+  //  { 4, 15, 14, 16 },
+  { 5, 17, 7, 8, 9, 18 },
+  { 6, 7, 17, 11, 13, 12 },
+  { 10, 9, 18, 4, 16, 14 },
+  
+  { 9, 16, 13, 12, 20 },
+  { 7, 13, 16, 14, 20 },
+  { 7, 14, 18, 20 },
+  { 9, 12, 17, 20 },
+  { 17, 7, 18, 9, 20 },
+  { 7, 17, 12, 13, 20 },
+  { 9, 18, 14, 16, 20 },
+};
+HPRef_Struct reftet_3ec_1v =
+{
+  HP_TET,
+  reftet_3ec_1v_splitedges, 
+  reftet_3ec_1v_splitfaces, 
+  reftet_3ec_1v_splitelements, 
+  reftet_3ec_1v_newelstypes, 
+  reftet_3ec_1v_newels
+};
+
+
+
+
+
+
+
+
+//  HP_TET_3EC_2V,  
+int reftet_3ec_2v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 2, 3, 9 },
+  { 2, 4, 10 },
+  { 3, 1, 11 },
+  { 3, 2, 12 },
+  { 3, 4, 13 },
+  { 4, 1, 14 },
+  { 4, 2, 15 },
+  { 4, 3, 16 },
+  { 0, 0, 0 }
+};
+int reftet_3ec_2v_splitfaces[][4] =
+  {
+    { 1, 2, 3, 17 },
+    { 2, 1, 4, 18 },
+    { 0, 0, 0, 0 }
+  };
+int reftet_3ec_2v_splitelements[][5] =
+  {
+    { 1, 2, 3, 4, 20 },
+    { 0 },
+  };
+
+HPREF_ELEMENT_TYPE reftet_3ec_2v_newelstypes[] =
+  {
+    HP_PYRAMID_EDGES,
+    HP_PYRAMID_EDGES,
+    HP_TET_1E_1VA,
+    HP_TET_1E_1VA,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    HP_PRISM_SINGEDGE,
+    
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_TET,
+    HP_TET,
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_PYRAMID,
+    HP_NONE,
+  };
+int reftet_3ec_2v_newels[][8] =
+{
+  { 1, 5, 17, 6, 7 },
+  { 2, 8, 18, 10, 9 },
+  { 3, 11, 12, 13 },
+  { 4, 15, 14, 16 },
+  { 5, 17, 7, 8, 9, 18 },
+  { 6, 7, 17, 11, 13, 12 },
+  { 10, 9, 18, 15, 16, 14 },
+  
+  { 9, 16, 13, 12, 20 },
+  { 7, 13, 16, 14, 20 },
+  { 7, 14, 18, 20 },
+  { 9, 12, 17, 20 },
+  { 17, 7, 18, 9, 20 },
+  { 7, 17, 12, 13, 20 },
+  { 9, 18, 14, 16, 20 },
+};
+HPRef_Struct reftet_3ec_2v =
+{
+  HP_TET,
+  reftet_3ec_2v_splitedges, 
+  reftet_3ec_2v_splitfaces, 
+  reftet_3ec_2v_splitelements, 
+  reftet_3ec_2v_newelstypes, 
+  reftet_3ec_2v_newels
+};
+
+
+
+
+
+
+
+
+
+
+/* ************************ 1 singular face ******************** */
+
+
+// HP_TET_1F_0E_0V
+int reftet_1f_0e_0v_splitedges[][3] =
+{
+  { 2, 1, 5 },
+  { 3, 1, 6 },
+  { 4, 1, 7 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftet_1f_0e_0v_newelstypes[] =
+{
+  HP_PRISM_1FA_0E_0V,
+  HP_TET,
+  HP_NONE,
+};
+int reftet_1f_0e_0v_newels[][8] =
+{
+  { 3, 2, 4, 6, 5, 7 },
+  { 5, 7, 6, 1 }
+};
+HPRef_Struct reftet_1f_0e_0v =
+{
+  HP_TET,
+  reftet_1f_0e_0v_splitedges, 
+  0, 0,
+  reftet_1f_0e_0v_newelstypes, 
+  reftet_1f_0e_0v_newels
+};
+
+
+
+
+
+// HP_TET_1F_0E_1VA    ... singular vertex in face
+int reftet_1f_0e_1va_splitedges[][3] =
+{
+  { 2, 1, 5 },
+  { 2, 3, 6 },
+  { 2, 4, 7 },
+  { 3, 1, 8 },
+  { 4, 1, 9 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftet_1f_0e_1va_newelstypes[] =
+{
+  HP_HEX_1F_0E_0V,
+  HP_TET_1F_0E_1VA,
+  HP_TET,
+  HP_NONE,
+};
+int reftet_1f_0e_1va_newels[][8] =
+{
+  { 3, 6, 7, 4, 8, 5, 5, 9 },
+  { 5, 2, 6, 7 },
+  { 5, 9, 8, 1 },
+};
+HPRef_Struct reftet_1f_0e_1va =
+{
+  HP_TET,
+  reftet_1f_0e_1va_splitedges, 
+  0, 0,
+  reftet_1f_0e_1va_newelstypes, 
+  reftet_1f_0e_1va_newels
+};
+
+
+
+
+
+// HP_TET_1F_0E_1VB    ... singular vertex not in face
+int reftet_1f_0e_1vb_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 1, 3, 6 },
+  { 1, 4, 7 },
+  { 2, 1, 8 },
+  { 3, 1, 9 },
+  { 4, 1, 10 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftet_1f_0e_1vb_newelstypes[] =
+{
+  HP_PRISM_1FA_0E_0V,
+  HP_PRISM,
+  HP_TET_0E_1V,
+  HP_NONE,
+};
+int reftet_1f_0e_1vb_newels[][8] =
+{
+  { 2, 4, 3, 8, 10, 9 },
+  { 8, 10, 9, 5, 7, 6 }, 
+  { 1, 5, 6, 7 },
+};
+HPRef_Struct reftet_1f_0e_1vb =
+{
+  HP_TET,
+  reftet_1f_0e_1vb_splitedges, 
+  0, 0,
+  reftet_1f_0e_1vb_newelstypes, 
+  reftet_1f_0e_1vb_newels
+};
+
+
+
+
+
+
+
+
+// HP_TET_1F_1EA_0V  ... sing edge is 1..2
+int reftet_1f_1ea_0v_splitedges[][3] =
+{
+  { 1, 3, 5 },
+  { 1, 4, 6 },
+  { 2, 1, 7 },
+  { 2, 3, 8 },
+  { 2, 4, 9 },
+  { 3, 1, 10 },
+  { 4, 1, 11 },
+  { 0, 0, 0 }
+};
+
+int reftet_1f_1ea_0v_splitfaces[][4] =
+  {
+    { 2, 1, 3, 12 },
+    { 2, 1, 4, 13 },
+    { 0, 0, 0, 0 }
+  };
+
+
+HPREF_ELEMENT_TYPE reftet_1f_1ea_0v_newelstypes[] =
+{
+  HP_HEX_1F_0E_0V,
+  //  HP_PRISM,
+  HP_PYRAMID_1FB_0E_1VA,
+  HP_TET_1E_1VA,
+  HP_PRISM_SINGEDGE,
+  HP_PRISM,
+  HP_NONE,
+};
+int reftet_1f_1ea_0v_newels[][8] =
+{
+  { 3, 8, 9, 4, 10, 12, 13, 11 },
+  // { 2, 9, 8, 7, 13, 12 },
+  { 8, 9, 13, 12, 2 },
+  { 2, 7, 13, 12 },
+  { 7, 13, 12, 1, 6, 5 },
+  { 6, 11, 13, 5, 10, 12 }
+};
+HPRef_Struct reftet_1f_1ea_0v =
+{
+  HP_TET,
+  reftet_1f_1ea_0v_splitedges, 
+  reftet_1f_1ea_0v_splitfaces, 
+  0, 
+  reftet_1f_1ea_0v_newelstypes, 
+  reftet_1f_1ea_0v_newels
+};
+
+
+
+
+
+
+
+
+// HP_TET_1F_1EB_0V     singular edge in face, edge is 2-3
+int reftet_1f_1eb_0v_splitedges[][3] =
+{
+  { 2, 1, 5 },
+  { 2, 4, 6 },
+  { 3, 1, 7 },
+  { 3, 4, 8 },
+  { 4, 1, 9 },
+  { 0, 0, 0 }
+};
+
+
+HPREF_ELEMENT_TYPE reftet_1f_1eb_0v_newelstypes[] =
+{
+  HP_PRISM_1FB_1EA_0V,
+  HP_PRISM_1FA_0E_0V,
+  HP_TET,
+  HP_NONE,
+};
+int reftet_1f_1eb_0v_newels[][8] =
+{
+  // { 2, 5, 6, 3, 7, 8 },
+  { 3, 8, 7, 2, 6, 5 },
+  { 6, 4, 8, 5, 9, 7 },
+  { 5, 9, 7, 1}
+};
+HPRef_Struct reftet_1f_1eb_0v =
+{
+  HP_TET,
+  reftet_1f_1eb_0v_splitedges, 
+  0, 0, 
+  reftet_1f_1eb_0v_newelstypes, 
+  reftet_1f_1eb_0v_newels
+};
+
+
+
+
+
+
+
+
+
+
+/* ************************ 2 singular faces ******************** */
+
+
+// HP_TET_2F_0E_0V
+int reftet_2f_0e_0v_splitedges[][3] =
+{
+  { 1, 2, 5 },
+  { 2, 1, 6 },
+  { 3, 1, 7 },
+  { 3, 2, 8 },
+  { 4, 1, 9 },
+  { 4, 2, 10 },
+  { 0, 0, 0 }
+};
+
+int reftet_2f_0e_0v_splitfaces[][4] =
+  {
+    { 3, 1, 2, 11 },
+    { 4, 1, 2, 12 },
+    { 0, 0, 0, 0 }
+  };
+
+
+HPREF_ELEMENT_TYPE reftet_2f_0e_0v_newelstypes[] =
+{
+  HP_PRISM_1FA_0E_0V,
+  HP_PRISM_1FA_0E_0V,
+  HP_PRISM_1FB_1EA_0V,
+  HP_PRISM_1FB_1EA_0V,
+  HP_TET,
+  HP_NONE,
+};
+int reftet_2f_0e_0v_newels[][8] =
+{
+  { 2, 10, 8, 6, 12, 11 },
+  { 1, 7, 9, 5, 11, 12 },
+  //   { 3, 11, 8, 4, 12, 10 },
+  { 4, 10, 12, 3, 8, 11 }, 
+  { 3, 7, 11, 4, 9, 12 },
+  { 5, 6, 11, 12 }
+};
+HPRef_Struct reftet_2f_0e_0v =
+{
+  HP_TET,
+  reftet_2f_0e_0v_splitedges, 
+  reftet_2f_0e_0v_splitfaces, 
+  0, 
+  reftet_2f_0e_0v_newelstypes, 
+  reftet_2f_0e_0v_newels
+};
+
+
diff --git a/Netgen/libsrc/meshing/hpref_trig.hpp b/Netgen/libsrc/meshing/hpref_trig.hpp
new file mode 100644
index 0000000000..6e2ede0eb0
--- /dev/null
+++ b/Netgen/libsrc/meshing/hpref_trig.hpp
@@ -0,0 +1,750 @@
+
+
+
+
+
+// HP_TRIG
+int reftrig_splitedges[][3] =
+{
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_newelstypes[] =
+{
+  HP_TRIG,
+  HP_NONE,
+};
+int reftrig_newels[][8] =
+{
+  { 1, 2, 3 },
+};
+HPRef_Struct reftrig =
+{
+  HP_TRIG, 
+  reftrig_splitedges, 
+  0, 0, 
+  reftrig_newelstypes, 
+  reftrig_newels
+};
+
+
+
+
+
+
+// HP_TRIG_SINGCORNER
+int reftrig_singcorner_splitedges[][3] =
+{
+  { 1, 2, 4 },
+  { 1, 3, 5 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_singcorner_newelstypes[] =
+{
+  HP_TRIG_SINGCORNER,
+  HP_QUAD,
+  HP_NONE,
+};
+int reftrig_singcorner_newels[][8] =
+{
+  { 1, 4, 5 },
+  { 2, 3, 5, 4 },
+};
+HPRef_Struct reftrig_singcorner =
+{
+  HP_TRIG,
+  reftrig_singcorner_splitedges, 
+  0, 0,
+  reftrig_singcorner_newelstypes, 
+  reftrig_singcorner_newels
+};
+
+
+/*
+// HP_TRIG_SINGCORNER, trigs only
+int reftrig_singcorner_splitedges[][3] =
+{
+  { 1, 2, 4 },
+  { 1, 3, 5 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_singcorner_newelstypes[] =
+{
+  HP_TRIG_SINGCORNER,
+  HP_TRIG,
+  HP_TRIG,
+  HP_NONE,
+};
+int reftrig_singcorner_newels[][8] =
+{
+  { 1, 4, 5 },
+  { 4, 2, 5 },
+  { 5, 2, 3 },
+};
+HPRef_Struct reftrig_singcorner =
+{
+  HP_TRIG,
+  reftrig_singcorner_splitedges, 
+  0, 0,
+  reftrig_singcorner_newelstypes, 
+  reftrig_singcorner_newels
+};
+*/
+
+
+
+
+
+// HP_TRIG_SINGCORNER12
+int reftrig_singcorner12_splitedges[][3] =
+{
+  { 1, 2, 4 },
+  { 1, 3, 5 },
+  { 2, 1, 6 },
+  { 2, 3, 7 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_singcorner12_newelstypes[] =
+{
+  HP_TRIG_SINGCORNER,
+  HP_TRIG_SINGCORNER,
+  HP_QUAD,
+  HP_TRIG,
+  HP_NONE,
+};
+int reftrig_singcorner12_newels[][8] =
+{
+  { 1, 4, 5 },
+  { 2, 7, 6 },
+  { 4, 6, 7, 5 },
+  { 5, 7, 3 },
+};
+HPRef_Struct reftrig_singcorner12 =
+{
+  HP_TRIG,
+  reftrig_singcorner12_splitedges, 
+  0, 0,
+  reftrig_singcorner12_newelstypes, 
+  reftrig_singcorner12_newels
+};
+
+
+
+
+// HP_TRIG_SINGCORNER123_2D
+int reftrig_singcorner123_2D_splitedges[][3] =
+{
+  { 1, 2, 4 },
+  { 1, 3, 5 },
+  { 2, 1, 6 },
+  { 2, 3, 7 },
+  { 3, 1, 8 },
+  { 3, 2, 9 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_singcorner123_2D_newelstypes[] =
+{
+  HP_TRIG_SINGCORNER,
+  HP_TRIG_SINGCORNER,
+  HP_TRIG_SINGCORNER,
+  HP_QUAD,
+  HP_QUAD,
+  HP_NONE,
+};
+int reftrig_singcorner123_2D_newels[][8] =
+{
+  { 1, 4, 5 },
+  { 2, 7, 6 },
+  { 3, 8, 9 },
+  { 4, 6, 8, 5 },
+  { 6, 7, 9, 8 },
+};
+HPRef_Struct reftrig_singcorner123_2D =
+{
+  HP_TRIG,
+  reftrig_singcorner123_2D_splitedges, 
+  0, 0,
+  reftrig_singcorner123_2D_newelstypes, 
+  reftrig_singcorner123_2D_newels
+};
+
+
+
+
+
+
+// HP_TRIG_SINGCORNER123
+int reftrig_singcorner123_splitedges[][3] =
+{
+  { 1, 2, 4 },
+  { 1, 3, 5 },
+  { 2, 1, 6 },
+  { 2, 3, 7 },
+  { 3, 1, 8 },
+  { 3, 2, 9 },
+  { 0, 0, 0 }
+};
+
+int reftrig_singcorner123_splitfaces[][4] =
+{
+  { 1, 2, 3, 10 },
+  { 2, 3, 1, 11 },
+  { 3, 1, 2, 12 },
+  { 0, 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_singcorner123_newelstypes[] =
+{
+  HP_DUMMY_QUAD_SINGCORNER,
+  HP_DUMMY_QUAD_SINGCORNER,
+  HP_DUMMY_QUAD_SINGCORNER,
+  //  HP_TRIG_SINGCORNER,
+  //  HP_TRIG,
+  //  HP_TRIG_SINGCORNER,
+  //  HP_TRIG,
+  //  HP_TRIG_SINGCORNER,
+  //  HP_TRIG,
+  HP_QUAD,
+  HP_QUAD,
+  HP_QUAD,
+  HP_TRIG,
+  HP_NONE,
+};
+int reftrig_singcorner123_newels[][8] =
+{
+  { 1, 4, 10, 5 },
+  { 2, 7, 11, 6 },
+  { 3, 8, 12, 9 },
+  //  { 1, 4, 5 },
+  //  { 5, 4, 10 },
+  //  { 2, 7, 6 },
+  //  { 6, 7, 11 },
+  //  { 3, 8, 9 },
+  //  { 8, 12, 9 },
+  { 4, 6, 11, 10 },
+  { 7, 9, 12, 11 },
+  { 8, 5, 10, 12 },
+  { 10, 11, 12 },
+};
+HPRef_Struct reftrig_singcorner123 =
+{
+  HP_TRIG,
+  reftrig_singcorner123_splitedges, 
+  reftrig_singcorner123_splitfaces, 
+  0, 
+  reftrig_singcorner123_newelstypes, 
+  reftrig_singcorner123_newels
+};
+
+
+
+
+
+
+
+
+
+
+
+
+// HP_TRIG_SINGEDGE
+int reftrig_singedge_splitedges[][3] =
+{
+  { 2, 3, 4 },
+  { 1, 3, 5 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_singedge_newelstypes[] =
+{
+  HP_TRIG,
+  HP_QUAD_SINGEDGE,
+  HP_NONE,
+};
+int reftrig_singedge_newels[][8] =
+{
+  { 4, 3, 5 },
+  { 1, 2, 4, 5 },
+};
+HPRef_Struct reftrig_singedge =
+{
+  HP_TRIG,
+  reftrig_singedge_splitedges, 
+  0, 0,
+  reftrig_singedge_newelstypes, 
+  reftrig_singedge_newels
+};
+
+
+
+
+
+
+// HP_TRIG_SINGEDGECORNER1
+int reftrig_singedgecorner1_splitedges[][3] =
+{
+  { 1, 2, 6 },
+  { 1, 3, 5 },
+  { 2, 3, 4 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_singedgecorner1_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG,
+  HP_NONE,
+};
+int reftrig_singedgecorner1_newels[][8] =
+{
+  { 1, 6, 5 },
+  { 6, 2, 4, 5 },
+  { 5, 4, 3 },
+};
+HPRef_Struct reftrig_singedgecorner1 =
+{
+  HP_TRIG,
+  reftrig_singedgecorner1_splitedges, 
+  0, 0, 
+  reftrig_singedgecorner1_newelstypes, 
+  reftrig_singedgecorner1_newels
+};
+
+
+
+
+
+
+
+
+// HP_TRIG_SINGEDGECORNER2
+int reftrig_singedgecorner2_splitedges[][3] =
+{
+  { 2, 1, 6 },
+  { 1, 3, 5 },
+  { 2, 3, 4 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_singedgecorner2_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG,
+  HP_NONE,
+};
+int reftrig_singedgecorner2_newels[][8] =
+{
+  { 6, 2, 4},
+  { 1, 6, 4, 5 },
+  { 5, 4, 3 },
+};
+HPRef_Struct reftrig_singedgecorner2 =
+{
+  HP_TRIG,
+  reftrig_singedgecorner2_splitedges, 
+  0, 0,
+  reftrig_singedgecorner2_newelstypes, 
+  reftrig_singedgecorner2_newels
+};
+
+
+
+
+// HP_TRIG_SINGEDGECORNER12
+int reftrig_singedgecorner12_splitedges[][3] =
+{
+  { 1, 2, 4 },
+  { 1, 3, 5 },
+  { 2, 1, 6 },
+  { 2, 3, 7 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_singedgecorner12_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG,
+  HP_NONE,
+};
+int reftrig_singedgecorner12_newels[][8] =
+{
+  { 1, 4, 5 },
+  { 6, 2, 7 },
+  { 4, 6, 7, 5 },
+  { 5, 7, 3 },
+};
+HPRef_Struct reftrig_singedgecorner12 =
+{
+  HP_TRIG,
+  reftrig_singedgecorner12_splitedges, 
+  0, 0,
+  reftrig_singedgecorner12_newelstypes, 
+  reftrig_singedgecorner12_newels
+};
+
+
+
+
+
+
+
+// HP_TRIG_SINGEDGECORNER3
+int reftrig_singedgecorner3_splitedges[][3] =
+{
+  { 1, 3, 4 },
+  { 3, 1, 5 },
+  { 2, 3, 6 },
+  { 3, 2, 7 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_singedgecorner3_newelstypes[] =
+{
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_NONE,
+};
+int reftrig_singedgecorner3_newels[][8] =
+{
+  { 1, 2, 6, 4 },
+  { 4, 6, 7, 5 },
+  { 3, 5, 7 },
+};
+HPRef_Struct reftrig_singedgecorner3 =
+{
+  HP_TRIG,
+  reftrig_singedgecorner3_splitedges, 
+  0, 0,
+  reftrig_singedgecorner3_newelstypes, 
+  reftrig_singedgecorner3_newels
+};
+
+
+
+
+// HP_TRIG_SINGEDGECORNER13
+int reftrig_singedgecorner13_splitedges[][3] =
+{
+  { 1, 2, 4 },
+  { 1, 3, 5 },
+  { 2, 3, 6 },
+  { 3, 1, 7 },
+  { 3, 2, 8 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_singedgecorner13_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_NONE,
+};
+int reftrig_singedgecorner13_newels[][8] =
+{
+  { 1, 4, 5 },
+  { 4, 2, 6, 5 },
+  { 5, 6, 8, 7 },
+  { 3, 7, 8 },
+};
+HPRef_Struct reftrig_singedgecorner13 =
+{
+  HP_TRIG,
+  reftrig_singedgecorner13_splitedges, 
+  0, 0,
+  reftrig_singedgecorner13_newelstypes, 
+  reftrig_singedgecorner13_newels
+};
+
+
+
+
+
+// HP_TRIG_SINGEDGECORNER23
+int reftrig_singedgecorner23_splitedges[][3] =
+{
+  { 1, 3, 4 },
+  { 2, 1, 5 },
+  { 2, 3, 6 },
+  { 3, 1, 7 },
+  { 3, 2, 8 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_singedgecorner23_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_NONE,
+};
+int reftrig_singedgecorner23_newels[][8] =
+{
+  { 5, 2, 6 },
+  { 1, 5, 6, 4 },
+  { 4, 6, 8, 7 },
+  { 3, 7, 8 },
+};
+HPRef_Struct reftrig_singedgecorner23 =
+{
+  HP_TRIG,
+  reftrig_singedgecorner23_splitedges, 
+  0, 0,
+  reftrig_singedgecorner23_newelstypes, 
+  reftrig_singedgecorner23_newels
+};
+
+
+
+// HP_TRIG_SINGEDGECORNER123
+int reftrig_singedgecorner123_splitedges[][3] =
+{
+  { 1, 2, 4 },
+  { 1, 3, 5 },
+  { 2, 1, 6 },
+  { 2, 3, 7 },
+  { 3, 1, 8 },
+  { 3, 2, 9 },
+  { 0, 0, 0 }
+};
+HPREF_ELEMENT_TYPE reftrig_singedgecorner123_newelstypes[] =
+{
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD,
+  HP_TRIG_SINGCORNER,
+  HP_NONE,
+};
+int reftrig_singedgecorner123_newels[][8] =
+{
+  { 1, 4, 5 },
+  { 6, 2, 7 },
+  { 4, 6, 7, 5 },
+  { 5, 7, 9, 8 },
+  { 3, 8, 9 },
+};
+HPRef_Struct reftrig_singedgecorner123 =
+{
+  HP_TRIG,
+  reftrig_singedgecorner123_splitedges, 
+  0, 0,
+  reftrig_singedgecorner123_newelstypes, 
+  reftrig_singedgecorner123_newels
+};
+
+
+
+
+
+
+
+
+
+
+
+
+
+// HP_TRIG_SINGEDGES
+int reftrig_singedges_splitedges[][3] =
+{
+  { 1, 2, 4 },
+  { 1, 3, 5 },
+  { 2, 3, 6 },
+  { 3, 2, 7 },
+  { 0, 0, 0 }
+};
+int reftrig_singedges_splitfaces[][4] =
+{
+  { 1, 2, 3, 8 },
+  { 0, 0, 0, 0 }
+};
+
+HPREF_ELEMENT_TYPE reftrig_singedges_newelstypes[] =
+{
+  //  HP_QUAD_2E,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG,
+  HP_NONE,
+};
+int reftrig_singedges_newels[][8] =
+{
+  // { 1, 4, 8, 5 },
+  { 1, 4, 8 },
+  { 5, 1, 8 },
+  { 4, 2, 6, 8 },
+  { 3, 5, 8, 7 },
+  { 6, 7, 8 },
+};
+HPRef_Struct reftrig_singedges =
+{
+  HP_TRIG,
+  reftrig_singedges_splitedges, 
+  reftrig_singedges_splitfaces, 
+  0,
+  reftrig_singedges_newelstypes, 
+  reftrig_singedges_newels
+};
+
+
+
+
+
+
+
+
+// HP_TRIG_SINGEDGES2
+int reftrig_singedges2_splitedges[][3] =
+{
+  { 1, 2, 4 },
+  { 1, 3, 5 },
+  { 2, 1, 6 },
+  { 2, 3, 7 },
+  { 3, 2, 8 },
+  { 0, 0, 0 }
+};
+int reftrig_singedges2_splitfaces[][4] =
+{
+  { 1, 2, 3, 9 },
+  { 0, 0, 0, 0 }
+};
+
+HPREF_ELEMENT_TYPE reftrig_singedges2_newelstypes[] =
+{
+  //  HP_QUAD_2E,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG,
+  HP_NONE,
+};
+int reftrig_singedges2_newels[][8] =
+{
+  //  { 1, 4, 9, 5 },
+  { 1, 4, 9 },
+  { 5, 1, 9 },
+  { 4, 6, 7, 9 },
+  { 3, 5, 9, 8 },
+  { 6, 2, 7 },
+  { 7, 8, 9 },
+};
+HPRef_Struct reftrig_singedges2 =
+{
+  HP_TRIG,
+  reftrig_singedges2_splitedges, 
+  reftrig_singedges2_splitfaces, 
+  0,
+  reftrig_singedges2_newelstypes, 
+  reftrig_singedges2_newels
+};
+
+
+
+
+// HP_TRIG_SINGEDGES3
+int reftrig_singedges3_splitedges[][3] =
+{
+  { 1, 2, 4 },
+  { 1, 3, 5 },
+  { 2, 3, 6 },
+  { 3, 1, 7 },
+  { 3, 2, 8 },
+  { 0, 0, 0 }
+};
+int reftrig_singedges3_splitfaces[][4] =
+{
+  { 1, 2, 3, 9 },
+  { 0, 0, 0, 0 }
+};
+
+HPREF_ELEMENT_TYPE reftrig_singedges3_newelstypes[] =
+{
+  //  HP_QUAD_2E,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG,
+  HP_NONE,
+};
+int reftrig_singedges3_newels[][8] =
+{
+  //  { 1, 4, 9, 5 },
+  { 1, 4, 9 },
+  { 5, 1, 9 },
+  { 4, 2, 6, 9 },
+  { 7, 5, 9, 8 },
+  { 3, 7, 8 },
+  { 6, 8, 9 },
+};
+HPRef_Struct reftrig_singedges3 =
+{
+  HP_TRIG,
+  reftrig_singedges3_splitedges, 
+  reftrig_singedges3_splitfaces, 
+  0,
+  reftrig_singedges3_newelstypes, 
+  reftrig_singedges3_newels
+};
+
+
+
+
+
+
+// HP_TRIG_SINGEDGES23
+int reftrig_singedges23_splitedges[][3] =
+{
+  { 1, 2, 4 },
+  { 1, 3, 5 },
+  { 2, 1, 6 },
+  { 2, 3, 7 },
+  { 3, 1, 8 },
+  { 3, 2, 9 },
+  { 0, 0, 0 }
+};
+int reftrig_singedges23_splitfaces[][4] =
+{
+  { 1, 2, 3, 10 },
+  { 0, 0, 0, 0 }
+};
+
+HPREF_ELEMENT_TYPE reftrig_singedges23_newelstypes[] =
+{
+  //  HP_QUAD_2E,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG_SINGEDGECORNER2,
+
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_SINGEDGE,
+  HP_TRIG_SINGEDGECORNER2,
+  HP_TRIG_SINGEDGECORNER1,
+  HP_TRIG,
+  HP_NONE,
+};
+int reftrig_singedges23_newels[][8] =
+{
+  //  { 1, 4, 10, 5 },
+  { 1 , 4, 10 },
+  { 5, 1, 10 },
+  { 4, 6, 7, 10 },
+  { 8, 5, 10, 9 },
+  { 6, 2, 7 },
+  { 3, 8, 9 },
+  { 7, 9, 10 },
+};
+HPRef_Struct reftrig_singedges23 =
+{
+  HP_TRIG,
+  reftrig_singedges23_splitedges, 
+  reftrig_singedges23_splitfaces, 
+  0,
+  reftrig_singedges23_newelstypes, 
+  reftrig_singedges23_newels
+};
+
+
diff --git a/Netgen/libsrc/meshing/hprefinement.cpp b/Netgen/libsrc/meshing/hprefinement.cpp
index be3bbf20b2..6a63e87675 100644
--- a/Netgen/libsrc/meshing/hprefinement.cpp
+++ b/Netgen/libsrc/meshing/hprefinement.cpp
@@ -1,586 +1,457 @@
 #include <mystdlib.h>
 #include "meshing.hpp"
 
-namespace netgen
-{
-enum HPREF_ELEMENT_TYPE {
-  HP_NONE=0,
-  HP_TRIG = 10,
-  HP_TRIG_SINGCORNER,
-  HP_TRIG_SINGCORNER12,
-  HP_TRIG_SINGCORNER123,
-  HP_TRIG_SINGEDGE = 20,
-  HP_TRIG_SINGEDGECORNER1,
-  HP_TRIG_SINGEDGECORNER2,
-  HP_TRIG_SINGEDGECORNER12,
-  HP_TRIG_SINGEDGECORNER3,
-  HP_TRIG_SINGEDGECORNER13,
-  HP_TRIG_SINGEDGECORNER23,
-  HP_TRIG_SINGEDGECORNER123,
-  HP_TRIG_SINGEDGES = 30,
-  HP_TRIG_SINGEDGES2,
-  HP_TRIG_SINGEDGES3,
-  HP_TRIG_SINGEDGES23,
-  HP_TRIG_3SINGEDGES = 40,
-
-  HP_QUAD = 50,
-  HP_QUAD_SINGCORNER,
-  HP_DUMMY_QUAD_SINGCORNER,
-  HP_QUAD_SINGEDGE,
-  HP_QUAD_SINGEDGES,
-
-  HP_TET = 100,
-  HP_TET_0E_1V,
-  HP_TET_0E_2V,
-  HP_TET_0E_3V,
-  HP_TET_0E_4V,
-  HP_TET_1E_0V = 200,
-  HP_TET_1E_1VA,    // V on edge
-  HP_TET_1E_1VB,    // V not on edge
-  HP_TET_1E_2VA,    // both on edge
-  HP_TET_1E_2VB,
-  HP_TET_1E_2VC,
-  HP_TET_1E_2VD,    // non on edge
-  HP_TET_1E_3VA,    // 2 on edge
-  HP_TET_1E_3VB,    // 1 on edge
-  HP_TET_1E_4V,
-
-  // 2 connected edges, additonally marked Vs
-  HP_TET_2EA_0V = 220, 
-  HP_TET_2EA_1VA, 
-  HP_TET_2EA_1VB, 
-  HP_TET_2EA_1VC, 
-  HP_TET_2EA_2VA, 
-  HP_TET_2EA_2VB, 
-  HP_TET_2EA_2VC, 
-  HP_TET_2EA_3V, 
-
-  // 2 opposite edges
-  HP_TET_2EB_0V = 230,
-  HP_TET_2EB_1V,
-  HP_TET_2EB_2VA,
-  HP_TET_2EB_2VB,
-  HP_TET_2EB_2VC,
-  HP_TET_2EB_3V,
-  HP_TET_2EB_4V,
-
-  HP_TET_3EA_0V = 400,  // 3 edges connected
-  HP_TET_3EA_3V,
-
-  HP_TET_3EB_2V = 420,  // 3 edges chain
-  HP_TET_3EC_2V = 430,  // 3 edges chain, alter
-  HP_PRISM = 1000,
-  HP_PRISM_SINGEDGE,
-
-  HP_PYRAMID = 2000,
-  HP_PYRAMID_0E_1V,
-  HP_PYRAMID_EDGES,
-
-  HP_HEX = 3000,
-  HP_HEX_0E_1V,
-  HP_HEX_1E_0V,
-  HP_HEX_3E_0V,
-};
-
-
-struct HPRef_Struct {
-  HPREF_ELEMENT_TYPE geom;
-  int (*splitedges)[3];
-  int (*splitfaces)[4];
-  int (*splitelements)[5];
-  HPREF_ELEMENT_TYPE * neweltypes;
-  int (*newels)[8];
-};
-
-
-
-// HP_TRIG
-int reftrig_splitedges[][3] =
-{
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftrig_newelstypes[] =
-{
-  HP_TRIG,
-  HP_NONE,
-};
-int reftrig_newels[][8] =
-{
-  { 1, 2, 3 },
-};
-HPRef_Struct reftrig =
-{
-  HP_TRIG, 
-  reftrig_splitedges, 
-  0, 0, 
-  reftrig_newelstypes, 
-  reftrig_newels
-};
 
 
 
-
-// HP_TRIG_SINGCORNER
-int reftrig_singcorner_splitedges[][3] =
-{
-  { 1, 2, 4 },
-  { 1, 3, 5 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftrig_singcorner_newelstypes[] =
-{
-  HP_TRIG_SINGCORNER,
-  HP_QUAD,
-  HP_NONE,
-};
-int reftrig_singcorner_newels[][8] =
-{
-  { 1, 4, 5 },
-  { 2, 3, 5, 4 },
-};
-HPRef_Struct reftrig_singcorner =
+namespace netgen
 {
-  HP_TRIG,
-  reftrig_singcorner_splitedges, 
-  0, 0,
-  reftrig_singcorner_newelstypes, 
-  reftrig_singcorner_newels
-};
 
 
-/*
-// HP_TRIG_SINGCORNER, trigs only
-int reftrig_singcorner_splitedges[][3] =
-{
-  { 1, 2, 4 },
-  { 1, 3, 5 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftrig_singcorner_newelstypes[] =
-{
-  HP_TRIG_SINGCORNER,
-  HP_TRIG,
-  HP_TRIG,
-  HP_NONE,
-};
-int reftrig_singcorner_newels[][8] =
-{
-  { 1, 4, 5 },
-  { 4, 2, 5 },
-  { 5, 2, 3 },
-};
-HPRef_Struct reftrig_singcorner =
-{
-  HP_TRIG,
-  reftrig_singcorner_splitedges, 
-  0, 0,
-  reftrig_singcorner_newelstypes, 
-  reftrig_singcorner_newels
-};
-*/
+  // HP_SEGM
+  int refsegm_splitedges[][3] =
+    {
+      { 0, 0, 0 }
+    };
+  
+  HPREF_ELEMENT_TYPE refsegm_newelstypes[] =
+    {
+      HP_SEGM,
+      HP_NONE,
+    };
+  int refsegm_newels[][8] =
+    {
+      { 1, 2 },
+    };
+  HPRef_Struct refsegm =
+    {
+      HP_SEGM, 
+      refsegm_splitedges, 
+      0, 0, 
+      refsegm_newelstypes, 
+      refsegm_newels
+    };
 
 
+  // HP_SEGM_SINGCORNERL = 2,
+  int refsegm_scl_splitedges[][3] =
+    {
+      { 1, 2, 3 },
+      { 0, 0, 0 }
+    };
 
+  HPREF_ELEMENT_TYPE refsegm_scl_newelstypes[] =
+    {
+      HP_SEGM_SINGCORNERL,
+      HP_SEGM,
+      HP_NONE,
+    };
+  int refsegm_scl_newels[][8] =
+    {
+      { 1, 3 },
+      { 3, 2 },
+      { 0, 0 },
+    };
+  HPRef_Struct refsegm_scl =
+    {
+      HP_SEGM,
+      refsegm_scl_splitedges,
+      0, 0,
+      refsegm_scl_newelstypes,
+      refsegm_scl_newels
+    };
 
 
-// HP_TRIG_SINGCORNER12
-int reftrig_singcorner12_splitedges[][3] =
-{
-  { 1, 2, 4 },
-  { 1, 3, 5 },
-  { 2, 1, 6 },
-  { 2, 3, 7 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftrig_singcorner12_newelstypes[] =
-{
-  HP_TRIG_SINGCORNER,
-  HP_TRIG_SINGCORNER,
-  HP_QUAD,
-  HP_TRIG,
-  HP_NONE,
-};
-int reftrig_singcorner12_newels[][8] =
-{
-  { 1, 4, 5 },
-  { 2, 7, 6 },
-  { 4, 6, 7, 5 },
-  { 5, 7, 3 },
-};
-HPRef_Struct reftrig_singcorner12 =
-{
-  HP_TRIG,
-  reftrig_singcorner12_splitedges, 
-  0, 0,
-  reftrig_singcorner12_newelstypes, 
-  reftrig_singcorner12_newels
-};
 
+  // HP_SEGM_SINGCORNERR
+  int refsegm_scr_splitedges[][3] =
+    {
+      { 2, 1, 3 },
+      { 0, 0, 0 }
+    };
 
+  HPREF_ELEMENT_TYPE refsegm_scr_newelstypes[] =
+    {
+      HP_SEGM,
+      HP_SEGM_SINGCORNERR,
+      HP_NONE,
+    };
+  int refsegm_scr_newels[][8] =
+    {
+      { 1, 3 },
+      { 3, 2 },
+      { 0, 0 },
+    };
+  HPRef_Struct refsegm_scr =
+    {
+      HP_SEGM,
+      refsegm_scr_splitedges,
+      0, 0,
+      refsegm_scr_newelstypes,
+      refsegm_scr_newels
+    };
 
 
-// HP_TRIG_SINGCORNER123
-int reftrig_singcorner123_splitedges[][3] =
-{
-  { 1, 2, 4 },
-  { 1, 3, 5 },
-  { 2, 1, 6 },
-  { 2, 3, 7 },
-  { 3, 1, 8 },
-  { 3, 2, 9 },
-  { 0, 0, 0 }
-};
-int reftrig_singcorner123_splitfaces[][4] =
-{
-  { 1, 2, 3, 10 },
-  { 2, 3, 1, 11 },
-  { 3, 1, 2, 12 },
-  { 0, 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftrig_singcorner123_newelstypes[] =
-{
-  HP_DUMMY_QUAD_SINGCORNER,
-  HP_DUMMY_QUAD_SINGCORNER,
-  HP_DUMMY_QUAD_SINGCORNER,
-  //  HP_TRIG_SINGCORNER,
-  //  HP_TRIG,
-  //  HP_TRIG_SINGCORNER,
-  //  HP_TRIG,
-  //  HP_TRIG_SINGCORNER,
-  //  HP_TRIG,
-  HP_QUAD,
-  HP_QUAD,
-  HP_QUAD,
-  HP_TRIG,
-  HP_NONE,
-};
-int reftrig_singcorner123_newels[][8] =
-{
-  { 1, 4, 10, 5 },
-  { 2, 7, 11, 6 },
-  { 3, 8, 12, 9 },
-  //  { 1, 4, 5 },
-  //  { 5, 4, 10 },
-  //  { 2, 7, 6 },
-  //  { 6, 7, 11 },
-  //  { 3, 8, 9 },
-  //  { 8, 12, 9 },
-  { 4, 6, 11, 10 },
-  { 7, 9, 12, 11 },
-  { 8, 5, 10, 12 },
-  { 10, 11, 12 },
-};
-HPRef_Struct reftrig_singcorner123 =
-{
-  HP_TRIG,
-  reftrig_singcorner123_splitedges, 
-  reftrig_singcorner123_splitfaces, 
-  0, 
-  reftrig_singcorner123_newelstypes, 
-  reftrig_singcorner123_newels
-};
 
 
 
 
+  // HP_SEGM_SINGCORNERS = 3,
+  int refsegm_sc2_splitedges[][3] =
+    {
+      { 1, 2, 3 },
+      { 2, 1, 4 },
+      { 0, 0, 0 }
+    };
 
+  HPREF_ELEMENT_TYPE refsegm_sc2_newelstypes[] =
+    {
+      HP_SEGM_SINGCORNERL,
+      HP_SEGM_SINGCORNERR,
+      HP_SEGM,
+      HP_NONE,
+    };
+  int refsegm_sc2_newels[][8] =
+    {
+      { 1, 3 },
+      { 4, 2 },
+      { 3, 4 },
+      { 0, 0 },
+    };
+  HPRef_Struct refsegm_sc2 =
+    {
+      HP_SEGM,
+      refsegm_sc2_splitedges,
+      0, 0,
+      refsegm_sc2_newelstypes,
+      refsegm_sc2_newels
+    };
 
 
 
 
 
 
+#include "hpref_trig.hpp"
+#include "hpref_quad.hpp"
+#include "hpref_tet.hpp"
+#include "hpref_prism.hpp"
 
 
-// HP_TRIG_SINGEDGE
-int reftrig_singedge_splitedges[][3] =
-{
-  { 2, 3, 4 },
-  { 1, 3, 5 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftrig_singedge_newelstypes[] =
-{
-  HP_TRIG,
-  HP_QUAD_SINGEDGE,
-  HP_NONE,
-};
-int reftrig_singedge_newels[][8] =
-{
-  { 4, 3, 5 },
-  { 1, 2, 4, 5 },
-};
-HPRef_Struct reftrig_singedge =
-{
-  HP_TRIG,
-  reftrig_singedge_splitedges, 
-  0, 0,
-  reftrig_singedge_newelstypes, 
-  reftrig_singedge_newels
-};
 
 
 
+  // HP_PYRAMID
+  int refpyramid_splitedges[][3] =
+    {
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refpyramid_newelstypes[] =
+    {
+      HP_PYRAMID,
+      HP_NONE,
+    };
+  int refpyramid_newels[][8] =
+    {
+      { 1, 2, 3, 4, 5, 6 }
+    };
+  HPRef_Struct refpyramid =
+    {
+      HP_PYRAMID,
+      refpyramid_splitedges, 
+      0, 0,
+      refpyramid_newelstypes, 
+      refpyramid_newels
+    };
 
 
 
-// HP_TRIG_SINGEDGECORNER1
-int reftrig_singedgecorner1_splitedges[][3] =
-{
-  { 1, 2, 6 },
-  { 1, 3, 5 },
-  { 2, 3, 4 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftrig_singedgecorner1_newelstypes[] =
-{
-  HP_TRIG_SINGEDGECORNER1,
-  HP_QUAD_SINGEDGE,
-  HP_TRIG,
-  HP_NONE,
-};
-int reftrig_singedgecorner1_newels[][8] =
-{
-  { 1, 6, 5 },
-  { 6, 2, 4, 5 },
-  { 5, 4, 3 },
-};
-HPRef_Struct reftrig_singedgecorner1 =
-{
-  HP_TRIG,
-  reftrig_singedgecorner1_splitedges, 
-  0, 0, 
-  reftrig_singedgecorner1_newelstypes, 
-  reftrig_singedgecorner1_newels
-};
+  // HP_PYRAMID_0E_1V
+  int refpyramid_0e_1v_splitedges[][3] =
+    {
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refpyramid_0e_1v_newelstypes[] =
+    {
+      HP_TET_0E_1V,
+      HP_TET,
+      HP_NONE,
+    };
+  int refpyramid_0e_1v_newels[][8] =
+    {
+      { 1, 2, 4, 5 },
+      { 2, 3, 4, 5 },
+    };
+  HPRef_Struct refpyramid_0e_1v =
+    {
+      HP_PYRAMID,
+      refpyramid_0e_1v_splitedges, 
+      0, 0,
+      refpyramid_0e_1v_newelstypes, 
+      refpyramid_0e_1v_newels
+    };
 
 
 
+  // HP_PYRAMID_EDGES
+  int refpyramid_edges_splitedges[][3] =
+    {
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refpyramid_edges_newelstypes[] =
+    {
+      HP_TET_1E_1VA,
+      HP_TET_1E_1VA,
+      HP_NONE,
+    };
+  int refpyramid_edges_newels[][8] =
+    {
+      { 1, 2, 3, 5 },
+      { 1, 4, 5, 3 },
+    };
+  HPRef_Struct refpyramid_edges =
+    {
+      HP_PYRAMID,
+      refpyramid_edges_splitedges, 
+      0, 0,
+      refpyramid_edges_newelstypes, 
+      refpyramid_edges_newels
+    };
 
 
 
 
+  // HP_PYRAMID_1FB_0E_1VA
+  int refpyramid_1fb_0e_1va_splitedges[][3] =
+    {
+      { 1, 4, 6 },
+      { 2, 3, 7 },
+      { 5, 1, 8 },
+      { 5, 2, 9 },
+      { 5, 3, 10 },
+      { 5, 4, 11 },
+      { 0, 0, 0 },
+    };
+
+  HPREF_ELEMENT_TYPE refpyramid_1fb_0e_1va_newelstypes[] =
+    {
+      HP_HEX_1F_0E_0V,
+      HP_PYRAMID_1FB_0E_1VA,
+      HP_PRISM,
+      HP_NONE,
+    };
+  int refpyramid_1fb_0e_1va_newels[][8] =
+    {
+      { 1, 8, 9, 2, 6, 11, 10, 7 },
+      { 8, 9, 10, 11, 5 },
+      { 3, 7, 10, 4, 6, 11 }
+    };
+  HPRef_Struct refpyramid_1fb_0e_1va =
+    {
+      HP_PYRAMID,
+      refpyramid_1fb_0e_1va_splitedges, 
+      0, 0,
+      refpyramid_1fb_0e_1va_newelstypes, 
+      refpyramid_1fb_0e_1va_newels
+    };
 
-// HP_TRIG_SINGEDGECORNER2
-int reftrig_singedgecorner2_splitedges[][3] =
-{
-  { 2, 1, 6 },
-  { 1, 3, 5 },
-  { 2, 3, 4 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftrig_singedgecorner2_newelstypes[] =
-{
-  HP_TRIG_SINGEDGECORNER2,
-  HP_QUAD_SINGEDGE,
-  HP_TRIG,
-  HP_NONE,
-};
-int reftrig_singedgecorner2_newels[][8] =
-{
-  { 6, 2, 4},
-  { 1, 6, 4, 5 },
-  { 5, 4, 3 },
-};
-HPRef_Struct reftrig_singedgecorner2 =
-{
-  HP_TRIG,
-  reftrig_singedgecorner2_splitedges, 
-  0, 0,
-  reftrig_singedgecorner2_newelstypes, 
-  reftrig_singedgecorner2_newels
-};
 
 
 
 
-// HP_TRIG_SINGEDGECORNER12
-int reftrig_singedgecorner12_splitedges[][3] =
-{
-  { 1, 2, 4 },
-  { 1, 3, 5 },
-  { 2, 1, 6 },
-  { 2, 3, 7 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftrig_singedgecorner12_newelstypes[] =
-{
-  HP_TRIG_SINGEDGECORNER1,
-  HP_TRIG_SINGEDGECORNER2,
-  HP_QUAD_SINGEDGE,
-  HP_TRIG,
-  HP_NONE,
-};
-int reftrig_singedgecorner12_newels[][8] =
-{
-  { 1, 4, 5 },
-  { 6, 2, 7 },
-  { 4, 6, 7, 5 },
-  { 5, 7, 3 },
-};
-HPRef_Struct reftrig_singedgecorner12 =
-{
-  HP_TRIG,
-  reftrig_singedgecorner12_splitedges, 
-  0, 0,
-  reftrig_singedgecorner12_newelstypes, 
-  reftrig_singedgecorner12_newels
-};
 
 
+  // HP_HEX
+  int refhex_splitedges[][3] =
+    {
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refhex_newelstypes[] =
+    {
+      HP_HEX,
+      HP_NONE,
+    };
+  int refhex_newels[][8] =
+    {
+      { 1, 2, 3, 4, 5, 6, 7, 8 }
+    };
+  HPRef_Struct refhex =
+    {
+      HP_HEX,
+      refhex_splitedges, 
+      0, 0,
+      refhex_newelstypes, 
+      refhex_newels
+    };
 
 
 
 
+  // HP_HEX_0E_1V
+  int refhex_0e_1v_splitedges[][3] =
+    {
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refhex_0e_1v_newelstypes[] =
+    {
+      HP_TET_0E_1V,
+      HP_TET,
+      HP_TET,
+      HP_TET,
+      HP_TET,
+      HP_TET,
+      HP_NONE,
+    };
+  int refhex_0e_1v_newels[][8] =
+    {
+      { 1, 5, 2, 4 },
+      { 7, 3, 6, 8 },
+      { 2, 8, 5, 6 },
+      { 2, 8, 6, 3 },
+      { 2, 8, 3, 4 },
+      { 2, 8, 4, 5 },
+    };
+  HPRef_Struct refhex_0e_1v =
+    {
+      HP_HEX,
+      refhex_0e_1v_splitedges, 
+      0, 0,
+      refhex_0e_1v_newelstypes, 
+      refhex_0e_1v_newels
+    };
 
-// HP_TRIG_SINGEDGECORNER3
-int reftrig_singedgecorner3_splitedges[][3] =
-{
-  { 1, 3, 4 },
-  { 3, 1, 5 },
-  { 2, 3, 6 },
-  { 3, 2, 7 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftrig_singedgecorner3_newelstypes[] =
-{
-  HP_QUAD_SINGEDGE,
-  HP_QUAD,
-  HP_TRIG_SINGCORNER,
-  HP_NONE,
-};
-int reftrig_singedgecorner3_newels[][8] =
-{
-  { 1, 2, 6, 4 },
-  { 4, 6, 7, 5 },
-  { 3, 5, 7 },
-};
-HPRef_Struct reftrig_singedgecorner3 =
-{
-  HP_TRIG,
-  reftrig_singedgecorner3_splitedges, 
-  0, 0,
-  reftrig_singedgecorner3_newelstypes, 
-  reftrig_singedgecorner3_newels
-};
 
 
 
+  // HP_HEX_1E_1V
+  int refhex_1e_1v_splitedges[][3] =
+    {
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refhex_1e_1v_newelstypes[] =
+    {
+      HP_TET_1E_1VA,
+      HP_TET,
+      HP_TET_0E_1V,
+      HP_TET_0E_1V,
+      HP_TET_0E_1V,
+      HP_TET_0E_1V,
+      HP_NONE,
+    };
+  int refhex_1e_1v_newels[][8] =
+    {
+      // { 1, 5, 2, 4 }, 
+      { 1, 2, 4, 5 },
+      { 7, 3, 6, 8 },
+      { 2, 8, 5, 6 },
+      { 2, 8, 6, 3 },
+      { 2, 8, 3, 4 },
+      { 2, 8, 4, 5 },
+    };
+  HPRef_Struct refhex_1e_1v =
+    {
+      HP_HEX,
+      refhex_1e_1v_splitedges, 
+      0, 0,
+      refhex_1e_1v_newelstypes, 
+      refhex_1e_1v_newels
+    };
 
-// HP_TRIG_SINGEDGECORNER13
-int reftrig_singedgecorner13_splitedges[][3] =
-{
-  { 1, 2, 4 },
-  { 1, 3, 5 },
-  { 2, 3, 6 },
-  { 3, 1, 7 },
-  { 3, 2, 8 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftrig_singedgecorner13_newelstypes[] =
-{
-  HP_TRIG_SINGEDGECORNER1,
-  HP_QUAD_SINGEDGE,
-  HP_QUAD,
-  HP_TRIG_SINGCORNER,
-  HP_NONE,
-};
-int reftrig_singedgecorner13_newels[][8] =
-{
-  { 1, 4, 5 },
-  { 4, 2, 6, 5 },
-  { 5, 6, 8, 7 },
-  { 3, 7, 8 },
-};
-HPRef_Struct reftrig_singedgecorner13 =
-{
-  HP_TRIG,
-  reftrig_singedgecorner13_splitedges, 
-  0, 0,
-  reftrig_singedgecorner13_newelstypes, 
-  reftrig_singedgecorner13_newels
-};
 
 
+  // HP_HEX_3E_0V
+  int refhex_3e_0v_splitedges[][3] =
+    {
+      { 0, 0, 0 }
+    };
+  HPREF_ELEMENT_TYPE refhex_3e_0v_newelstypes[] =
+    {
+      HP_TET_1E_1VA,
+      HP_TET_1E_1VA,
+      HP_TET_1E_1VA,
+      HP_TET_0E_1V,
+      HP_TET,
+      HP_NONE,
+    };
+  int refhex_3e_0v_newels[][8] =
+    {
+      { 1, 2, 3, 6 },
+      { 1, 4, 8, 3 },
+      { 1, 5, 6, 8 },
+      { 1, 6, 3, 8 },
+      { 3, 8, 6, 7 },
+    };
+  HPRef_Struct refhex_3e_0v =
+    {
+      HP_HEX,
+      refhex_3e_0v_splitedges, 
+      0, 0,
+      refhex_3e_0v_newelstypes, 
+      refhex_3e_0v_newels
+    };
 
 
 
-// HP_TRIG_SINGEDGECORNER23
-int reftrig_singedgecorner23_splitedges[][3] =
-{
-  { 1, 3, 4 },
-  { 2, 1, 5 },
-  { 2, 3, 6 },
-  { 3, 1, 7 },
-  { 3, 2, 8 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftrig_singedgecorner23_newelstypes[] =
-{
-  HP_TRIG_SINGEDGECORNER2,
-  HP_QUAD_SINGEDGE,
-  HP_QUAD,
-  HP_TRIG_SINGCORNER,
-  HP_NONE,
-};
-int reftrig_singedgecorner23_newels[][8] =
-{
-  { 5, 2, 6 },
-  { 1, 5, 6, 4 },
-  { 4, 6, 8, 7 },
-  { 3, 7, 8 },
-};
-HPRef_Struct reftrig_singedgecorner23 =
-{
-  HP_TRIG,
-  reftrig_singedgecorner23_splitedges, 
-  0, 0,
-  reftrig_singedgecorner23_newelstypes, 
-  reftrig_singedgecorner23_newels
-};
 
+  // HP_HEX_1E_0V
+  int refhex_1e_0v_splitedges[][3] =
+    {
+      { 0, 0, 0 }
+    };
 
+  HPREF_ELEMENT_TYPE refhex_1e_0v_newelstypes[] =
+    {
+      HP_PRISM_SINGEDGE,
+      HP_PRISM,
+      HP_NONE,
+    };
+  int refhex_1e_0v_newels[][8] =
+    {
+      { 1, 4, 5, 2, 3, 6 },
+      { 5, 4, 8, 6, 3, 7 },
+    };
+  HPRef_Struct refhex_1e_0v =
+    {
+      HP_HEX,
+      refhex_1e_0v_splitedges, 
+      0, 0,
+      refhex_1e_0v_newelstypes, 
+      refhex_1e_0v_newels
+    };
 
-// HP_TRIG_SINGEDGECORNER123
-int reftrig_singedgecorner123_splitedges[][3] =
-{
-  { 1, 2, 4 },
-  { 1, 3, 5 },
-  { 2, 1, 6 },
-  { 2, 3, 7 },
-  { 3, 1, 8 },
-  { 3, 2, 9 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftrig_singedgecorner123_newelstypes[] =
-{
-  HP_TRIG_SINGEDGECORNER1,
-  HP_TRIG_SINGEDGECORNER2,
-  HP_QUAD_SINGEDGE,
-  HP_QUAD,
-  HP_TRIG_SINGCORNER,
-  HP_NONE,
-};
-int reftrig_singedgecorner123_newels[][8] =
-{
-  { 1, 4, 5 },
-  { 6, 2, 7 },
-  { 4, 6, 7, 5 },
-  { 5, 7, 9, 8 },
-  { 3, 8, 9 },
-};
-HPRef_Struct reftrig_singedgecorner123 =
-{
-  HP_TRIG,
-  reftrig_singedgecorner123_splitedges, 
-  0, 0,
-  reftrig_singedgecorner123_newelstypes, 
-  reftrig_singedgecorner123_newels
-};
 
 
 
 
+  // HP_HEX_1F_0E_0V
+  int refhex_1f_0e_0v_splitedges[][3] =
+    {
+      { 1, 5, 9 },
+      { 2, 6, 10 },
+      { 3, 7, 11 },
+      { 4, 8, 12 },
+      { 0, 0, 0 }
+    };
+
+  HPREF_ELEMENT_TYPE refhex_1f_0e_0v_newelstypes[] =
+    {
+      HP_HEX_1F_0E_0V,
+      HP_HEX,
+      HP_NONE,
+    };
+  int refhex_1f_0e_0v_newels[][8] =
+    {
+      { 1, 2, 3, 4, 9, 10, 11, 12 },
+      { 9, 10, 11, 12, 5, 6, 7, 8 }
+    };
+  HPRef_Struct refhex_1f_0e_0v =
+    {
+      HP_HEX,
+      refhex_1f_0e_0v_splitedges, 
+      0, 0,
+      refhex_1f_0e_0v_newelstypes, 
+      refhex_1f_0e_0v_newels
+    };
 
 
 
@@ -588,1037 +459,1991 @@ HPRef_Struct reftrig_singedgecorner123 =
 
 
 
+  HPRef_Struct * Get_HPRef_Struct (HPREF_ELEMENT_TYPE type)
+  {
+    HPRef_Struct * hps = NULL;
+
+    switch (type)
+      {
+      case HP_SEGM:
+	hps = &refsegm; break;
+      case HP_SEGM_SINGCORNERL:
+	hps = &refsegm_scl; break;
+      case HP_SEGM_SINGCORNERR:
+	hps = &refsegm_scr; break;
+      case HP_SEGM_SINGCORNERS:
+	hps = &refsegm_sc2; break;
+
+      case HP_TRIG:
+	hps = &reftrig; break;
+      case HP_TRIG_SINGCORNER:
+	hps = &reftrig_singcorner; break;
+      case HP_TRIG_SINGCORNER12:
+	hps = &reftrig_singcorner12; break;
+      case HP_TRIG_SINGCORNER123:
+	hps = &reftrig_singcorner123; break;
+      case HP_TRIG_SINGCORNER123_2D:
+	hps = &reftrig_singcorner123_2D; break;
+      case HP_TRIG_SINGEDGE:
+	hps = &reftrig_singedge; break;
+      case HP_TRIG_SINGEDGECORNER1:
+	hps = &reftrig_singedgecorner1; break;
+      case HP_TRIG_SINGEDGECORNER2:
+	hps = &reftrig_singedgecorner2; break;
+      case HP_TRIG_SINGEDGECORNER12:
+	hps = &reftrig_singedgecorner12; break;
+      case HP_TRIG_SINGEDGECORNER3:
+	hps = &reftrig_singedgecorner3; break;
+      case HP_TRIG_SINGEDGECORNER13:
+	hps = &reftrig_singedgecorner13; break;
+      case HP_TRIG_SINGEDGECORNER23:
+	hps = &reftrig_singedgecorner23; break;
+      case HP_TRIG_SINGEDGECORNER123:
+	hps = &reftrig_singedgecorner123; break;
+      case HP_TRIG_SINGEDGES:
+	hps = &reftrig_singedges; break;
+      case HP_TRIG_SINGEDGES2:
+	hps = &reftrig_singedges2; break;
+      case HP_TRIG_SINGEDGES3:
+	hps = &reftrig_singedges3; break;
+      case HP_TRIG_SINGEDGES23:
+	hps = &reftrig_singedges23; break;
+      case HP_QUAD:
+	hps = &refquad; break;
+      case HP_DUMMY_QUAD_SINGCORNER:
+	hps = &refdummyquad_singcorner; break;
+      case HP_QUAD_SINGCORNER:
+	hps = &refquad_singcorner; break;
+      case HP_QUAD_SINGEDGE:
+	hps = &refquad_singedge; break;
+
+      case HP_QUAD_0E_2VA:
+	hps = &refquad_0e_2va; break;
+      case HP_QUAD_0E_2VB:
+	hps = &refquad_0e_2vb; break;
+
+      case HP_QUAD_0E_3V:
+	hps = &refquad_0e_3v; break;
+      case HP_QUAD_0E_4V:
+	hps = &refquad_0e_4v; break;
+
+      case HP_QUAD_1E_1VA:
+	hps = &refquad_1e_1va; break;
+      case HP_QUAD_1E_1VB:
+	hps = &refquad_1e_1vb; break;
+      case HP_QUAD_1E_1VC:
+	hps = &refquad_1e_1vc; break;
+      case HP_QUAD_1E_1VD:
+	hps = &refquad_1e_1vd; break;
+
+      case HP_QUAD_1E_2VA:
+	hps = &refquad_1e_2va; break;
+      case HP_QUAD_1E_2VB:
+	hps = &refquad_1e_2vb; break;
+      case HP_QUAD_1E_2VC:
+	hps = &refquad_1e_2vc; break;
+      case HP_QUAD_1E_2VD:
+	hps = &refquad_1e_2vd; break;
+      case HP_QUAD_1E_2VE:
+	hps = &refquad_1e_2ve; break;
+      case HP_QUAD_1E_2VF:
+	hps = &refquad_1e_2vf; break;
+
+      case HP_QUAD_1E_3VA:
+	hps = &refquad_1e_3va; break;
+      case HP_QUAD_1E_3VB:
+	hps = &refquad_1e_3vb; break;
+      case HP_QUAD_1E_3VC:
+	hps = &refquad_1e_3vc; break;
+      case HP_QUAD_1E_3VD:
+	hps = &refquad_1e_3vd; break;
+      case HP_QUAD_1E_4V:
+	hps = &refquad_1e_4v; break;
+
+
+      case HP_QUAD_2E:
+	hps = &refquad_2e; break;
+      case HP_QUAD_2E_1VA:
+	hps = &refquad_2e_1va; break;
+      case HP_QUAD_2E_1VB:
+	hps = &refquad_2e_1vb; break;
+      case HP_QUAD_2E_1VC:
+	hps = &refquad_2e_1vc; break;
+      case HP_QUAD_2E_2VA:
+	hps = &refquad_2e_2va; break;
+      case HP_QUAD_2E_2VB:
+	hps = &refquad_2e_2vb; break;
+      case HP_QUAD_2E_2VC:
+	hps = &refquad_2e_2vc; break;
+      case HP_QUAD_2E_3V:
+	hps = &refquad_2e_3v; break;
+
+      case HP_QUAD_2EB_0V:
+	hps = &refquad_2eb_0v; break;
+
+      case HP_QUAD_2EB_1VA:
+	hps = &refquad_2eb_1va; break;
+      case HP_QUAD_2EB_1VB:
+	hps = &refquad_2eb_1vb; break;
+
+
+      case HP_QUAD_2EB_2VA:
+	hps = &refquad_2eb_2va; break;
+      case HP_QUAD_2EB_2VB:
+	hps = &refquad_2eb_2vb; break;
+      case HP_QUAD_2EB_2VC:
+	hps = &refquad_2eb_2vc; break;
+      case HP_QUAD_2EB_2VD:
+	hps = &refquad_2eb_2vd; break;
+
+      case HP_QUAD_2EB_3VA:
+	hps = &refquad_2eb_3va; break;
+      case HP_QUAD_2EB_3VB:
+	hps = &refquad_2eb_3vb; break;
+
+      case HP_QUAD_2EB_4V:
+	hps = &refquad_2eb_4v; break;
+
+      case HP_QUAD_3E:
+	hps = &refquad_3e; break;
+      case HP_QUAD_3E_3VA:
+	hps = &refquad_3e_3va; break;
+      case HP_QUAD_3E_3VB:
+	hps = &refquad_3e_3vb; break;
+      case HP_QUAD_3E_4V:
+	hps = &refquad_3e_4v; break;
+
+
+      case HP_QUAD_4E:
+	hps = &refquad_4e; break;
+
+
+      case HP_TET:
+	hps = &reftet; break;
+      case HP_TET_0E_1V:
+	hps = &reftet_0e_1v; break;
+      case HP_TET_0E_2V:
+	hps = &reftet_0e_2v; break;
+      case HP_TET_0E_3V:
+	hps = &reftet_0e_3v; break;
+      case HP_TET_0E_4V:
+	hps = &reftet_0e_4v; break;
+
+      case HP_TET_1E_0V:      
+	hps = &reftet_1e_0v; break;
+      case HP_TET_1E_1VA:
+	hps = &reftet_1e_1va; break;
+      case HP_TET_1E_1VB:
+	hps = &reftet_1e_1vb; break;
+
+      case HP_TET_1E_2VA:
+	hps = &reftet_1e_2va; break;
+      case HP_TET_1E_2VB:
+	hps = &reftet_1e_2vb; break;
+      case HP_TET_1E_2VC:
+	hps = &reftet_1e_2vc; break;
+      case HP_TET_1E_2VD:
+	hps = &reftet_1e_2vd; break;
+
+      case HP_TET_1E_3VA:
+	hps = &reftet_1e_3va; break;
+      case HP_TET_1E_3VB:
+	hps = &reftet_1e_3vb; break;
+      case HP_TET_1E_4V:
+	hps = &reftet_1e_4v; break;
+
+      case HP_TET_2EA_0V:
+	hps = &reftet_2ea_0v; break;
+      case HP_TET_2EA_1VB:
+	hps = &reftet_2ea_1vb; break;
+      case HP_TET_2EA_1VC:
+	hps = &reftet_2ea_1vc; break;
+      case HP_TET_2EA_1VA:
+	hps = &reftet_2ea_1va; break;
+      case HP_TET_2EA_2VA:
+	hps = &reftet_2ea_2va; break;
+      case HP_TET_2EA_2VB:
+	hps = &reftet_2ea_2vb; break;
+      case HP_TET_2EA_2VC:
+	hps = &reftet_2ea_2vc; break;
+      case HP_TET_2EA_3V:
+	hps = &reftet_2ea_3v; break;
+
+      case HP_TET_2EB_0V:
+	hps = &reftet_2eb_0v; break;
+      case HP_TET_2EB_2VA:
+	hps = &reftet_2eb_2va; break;
+      case HP_TET_2EB_4V:
+	hps = &reftet_2eb_4v; break;
+
+
+      case HP_TET_3EA_0V:
+	hps = &reftet_3ea_0v; break;
+      case HP_TET_3EA_1V:
+	hps = &reftet_3ea_1v; break;
+      case HP_TET_3EA_2V:
+	hps = &reftet_3ea_2v; break;
+      case HP_TET_3EA_3V:
+	hps = &reftet_3ea_3v; break;
+
+      case HP_TET_3EB_0V:
+	hps = &reftet_3eb_0v; break;
+      case HP_TET_3EB_1V:
+	hps = &reftet_3eb_1v; break;
+      case HP_TET_3EB_2V:
+	hps = &reftet_3eb_2v; break;
+      case HP_TET_3EC_0V:
+	hps = &reftet_3ec_0v; break;
+      case HP_TET_3EC_1V:
+	hps = &reftet_3ec_1v; break;
+      case HP_TET_3EC_2V:
+	hps = &reftet_3ec_2v; break;
+
+
+      case HP_TET_1F_0E_0V:
+	hps = &reftet_1f_0e_0v; break;
+      case HP_TET_1F_0E_1VA:
+	hps = &reftet_1f_0e_1va; break;
+      case HP_TET_1F_0E_1VB:
+	hps = &reftet_1f_0e_1vb; break;
+      case HP_TET_1F_1EA_0V:
+	hps = &reftet_1f_1ea_0v; break;
+      case HP_TET_1F_1EB_0V:
+	hps = &reftet_1f_1eb_0v; break;
+      case HP_TET_2F_0E_0V:
+	hps = &reftet_2f_0e_0v; break;
+
+
+      case HP_PRISM:
+	hps = &refprism; break;
+      case HP_PRISM_SINGEDGE:
+	hps = &refprism_singedge; break;
+      case HP_PRISM_SINGEDGE_H1:
+	hps = &refprism_singedge_h1; break;
+      case HP_PRISM_SINGEDGE_H12:
+	hps = &refprism_singedge_h12; break;
+      case HP_PRISM_SINGEDGE_V12:
+	hps = &refprism_singedge_v12; break;
+
+      case HP_PRISM_1FA_0E_0V:
+	hps = &refprism_1fa_0e_0v; break;
+      case HP_PRISM_1FB_0E_0V:
+	hps = &refprism_1fb_0e_0v; break;
+      case HP_PRISM_1FB_1EA_0V:
+	hps = &refprism_1fb_1ea_0v; break;
+
+      case HP_PYRAMID:
+	hps = &refpyramid; break;
+      case HP_PYRAMID_0E_1V:
+	hps = &refpyramid_0e_1v; break;
+      case HP_PYRAMID_EDGES:
+	hps = &refpyramid_edges; break;
+      case HP_PYRAMID_1FB_0E_1VA:
+	hps = &refpyramid_1fb_0e_1va; break;
+      case HP_HEX:
+	hps = &refhex; break;
+      case HP_HEX_0E_1V:
+	hps = &refhex_0e_1v; break;
+      case HP_HEX_1E_1V:
+	hps = &refhex_1e_1v; break;
+      case HP_HEX_1E_0V:
+	hps = &refhex_1e_0v; break;
+      case HP_HEX_3E_0V:
+	hps = &refhex_3e_0v; break;
+
+      case HP_HEX_1F_0E_0V:
+	hps = &refhex_1f_0e_0v; break;
+      }
+
+    if (!hps)
+      {
+	PrintSysError ("hp-refinement not implemented for case ", type);
+      }
+
+    return hps;
+  }
+
+
+  /* *********************** PrepareElements ****************************** */
+  
+  // Elements (volume, surface, edges)  are classified by singular vertices, edges, faces
 
 
-// HP_TRIG_SINGEDGES
-int reftrig_singedges_splitedges[][3] =
-{
-  { 1, 2, 4 },
-  { 1, 3, 5 },
-  { 2, 3, 6 },
-  { 3, 2, 7 },
-  { 0, 0, 0 }
-};
-int reftrig_singedges_splitfaces[][4] =
-{
-  { 1, 2, 3, 8 },
-  { 0, 0, 0, 0 }
-};
+  void PrepareElements (Mesh & mesh, ARRAY<HPRefElement> & elements)
+  {
+    INDEX_2_HASHTABLE<int> edges(mesh.GetNSeg()+1);
+    BitArray edgepoint(mesh.GetNP());
+    INDEX_2_HASHTABLE<int> edgepoint_dom(mesh.GetNSeg()+1);
 
-HPREF_ELEMENT_TYPE reftrig_singedges_newelstypes[] =
-{
-  //  HP_QUAD_SINGEDGES,
-  HP_TRIG_SINGEDGECORNER1,
-  HP_TRIG_SINGEDGECORNER2,
-  HP_QUAD_SINGEDGE,
-  HP_QUAD_SINGEDGE,
-  HP_TRIG,
-  HP_NONE,
-};
-int reftrig_singedges_newels[][8] =
-{
-  // { 1, 4, 8, 5 },
-  { 1, 4, 8 },
-  { 5, 1, 8 },
-  { 4, 2, 6, 8 },
-  { 3, 5, 8, 7 },
-  { 6, 7, 8 },
-};
-HPRef_Struct reftrig_singedges =
-{
-  HP_TRIG,
-  reftrig_singedges_splitedges, 
-  reftrig_singedges_splitfaces, 
-  0,
-  reftrig_singedges_newelstypes, 
-  reftrig_singedges_newels
-};
+    edgepoint.Clear();
+    BitArray cornerpoint(mesh.GetNP());
+    cornerpoint.Clear();
 
 
+    // value = nr > 0 ... refine elements in domain nr
+    // value = -1   ..... refine elements in any domain
+    INDEX_3_HASHTABLE<int> faces(mesh.GetNSE()+1);
+    INDEX_2_HASHTABLE<int> face_edges(mesh.GetNSE()+1);
+    ARRAY<int, PointIndex::BASE> facepoint(mesh.GetNP());
 
+    if (mesh.GetDimension() == 3)
+      {
+	/*
+	// check, if point has as least 3 different surfs:
 
+	ARRAY<INDEX_3, PointIndex::BASE> surfonpoint(mesh.GetNP());
+	surfonpoint = INDEX_3(0,0,0);
 
+	for (SurfaceElementIndex sei = 0; sei < mesh.GetNSE(); sei++)
+	  {
+	    const Element2d & el = mesh[sei];
+	    int ind = el.GetIndex();
+	    for (int j = 0; j < el.GetNP(); j++)
+	      {
+		INDEX_3 & i3 = surfonpoint[el[j]];
+		if (ind != i3.I1() && ind != i3.I2() && ind != i3.I3())
+		  {
+		    i3.I1() = i3.I2();
+		    i3.I2() = i3.I3();
+		    i3.I3() = ind;
+		  }
+	      }
+	  }
+	for (int i = 1; i <= mesh.GetNP(); i++)
+	  if (surfonpoint.Get(i).I1())
+	    cornerpoint.Set(i);
+	*/
+	cornerpoint.Clear();
 
+	for (int i = 1; i <= mesh.GetNP(); i++)
+	  {
+	    if (mesh.Point(i).IsSingular())
+	      cornerpoint.Set(i);
+	  }
 
 
-// HP_TRIG_SINGEDGES2
-int reftrig_singedges2_splitedges[][3] =
-{
-  { 1, 2, 4 },
-  { 1, 3, 5 },
-  { 2, 1, 6 },
-  { 2, 3, 7 },
-  { 3, 2, 8 },
-  { 0, 0, 0 }
-};
-int reftrig_singedges2_splitfaces[][4] =
-{
-  { 1, 2, 3, 9 },
-  { 0, 0, 0, 0 }
-};
+	for (int i = 1; i <= mesh.GetNSeg(); i++)
+	  if (mesh.LineSegment(i).singedge_left)
+	    {
+	      INDEX_2 i2 (mesh.LineSegment(i).p1, 
+			  mesh.LineSegment(i).p2);
+	      i2.Sort();
+	      
+	      edges.Set (i2, 1);
+	      INDEX_2 i2s(i2.I2(), i2.I1());
+	      edges.Set (i2s, 1);
+	      
+	      edgepoint.Set (i2.I1());
+	      edgepoint.Set (i2.I2());
+	    }
 
-HPREF_ELEMENT_TYPE reftrig_singedges2_newelstypes[] =
-{
-  //  HP_QUAD_SINGEDGES,
-  HP_TRIG_SINGEDGECORNER1,
-  HP_TRIG_SINGEDGECORNER2,
-  HP_QUAD_SINGEDGE,
-  HP_QUAD_SINGEDGE,
-  HP_TRIG_SINGEDGECORNER2,
-  HP_TRIG,
-  HP_NONE,
-};
-int reftrig_singedges2_newels[][8] =
-{
-  //  { 1, 4, 9, 5 },
-  { 1, 4, 9 },
-  { 5, 1, 9 },
-  { 4, 6, 7, 9 },
-  { 3, 5, 9, 8 },
-  { 6, 2, 7 },
-  { 7, 8, 9 },
-};
-HPRef_Struct reftrig_singedges2 =
-{
-  HP_TRIG,
-  reftrig_singedges2_splitedges, 
-  reftrig_singedges2_splitfaces, 
-  0,
-  reftrig_singedges2_newelstypes, 
-  reftrig_singedges2_newels
-};
 
+	// if 2 adjacent edges of an element are singular, the 
+	// commen point must be a singular point
+	for (int i = 1; i <= mesh.GetNE(); i++)
+	  {
+	    const Element & el = mesh.VolumeElement(i);
+	    const ELEMENT_EDGE * eledges = MeshTopology::GetEdges (el.GetType());
+	    int nedges = MeshTopology::GetNEdges (el.GetType());
+	    for (int j = 0; j < nedges; j++)
+	      for (int k = 0; k < nedges; k++)
+		if (j != k)
+		  {
+		    INDEX_2 ej(el.PNum(eledges[j][0]), el.PNum(eledges[j][1]));
+		    ej.Sort();
+		    INDEX_2 ek(el.PNum(eledges[k][0]), el.PNum(eledges[k][1]));
+		    ek.Sort();
+		    if (edges.Used(ej) && edges.Used(ek))
+		      {
+			if (ej.I1() == ek.I1()) cornerpoint.Set (ek.I1());
+			if (ej.I1() == ek.I2()) cornerpoint.Set (ek.I2());
+			if (ej.I2() == ek.I1()) cornerpoint.Set (ek.I1());
+			if (ej.I2() == ek.I2()) cornerpoint.Set (ek.I2());
+		      }
+		  }
+	  }
 
+	
+	edgepoint.Or (cornerpoint);
+	
+	(*testout) << "cornerpoint = " << endl << cornerpoint << endl;
+	(*testout) << "edgepoint = " << endl << edgepoint << endl;
 
+	facepoint = 0;
+	for (SurfaceElementIndex sei = 0; sei < mesh.GetNSE(); sei++)
+	  {
+	    const Element2d & el = mesh[sei];
+	    const FaceDescriptor & fd = mesh.GetFaceDescriptor (el.GetIndex());
+
+	    if (!fd.domin_singular && !fd.domout_singular) continue;
+	    int domnr;
+	    if (fd.domin_singular) domnr = fd.DomainIn();
+	    if (fd.domout_singular) domnr = fd.DomainOut();
+	    if (fd.domin_singular && fd.domout_singular) domnr = -1;
+	    
 
-// HP_TRIG_SINGEDGES3
-int reftrig_singedges3_splitedges[][3] =
-{
-  { 1, 2, 4 },
-  { 1, 3, 5 },
-  { 2, 3, 6 },
-  { 3, 1, 7 },
-  { 3, 2, 8 },
-  { 0, 0, 0 }
-};
-int reftrig_singedges3_splitfaces[][4] =
-{
-  { 1, 2, 3, 9 },
-  { 0, 0, 0, 0 }
-};
+	    faces.Set (INDEX_3::Sort (el[0], el[1], el[2]), domnr);
+	    face_edges.Set (INDEX_2::Sort (el[0], el[1]), domnr);
+	    face_edges.Set (INDEX_2::Sort (el[0], el[2]), domnr);
+	    face_edges.Set (INDEX_2::Sort (el[2], el[1]), domnr);
+	    facepoint[el[0]] = domnr;
+	    facepoint[el[1]] = domnr;
+	    facepoint[el[2]] = domnr;
+	  }
+	
+      }
+    else
+      {
+	// 2D case
 
-HPREF_ELEMENT_TYPE reftrig_singedges3_newelstypes[] =
-{
-  //  HP_QUAD_SINGEDGES,
-  HP_TRIG_SINGEDGECORNER1,
-  HP_TRIG_SINGEDGECORNER2,
-  HP_QUAD_SINGEDGE,
-  HP_QUAD_SINGEDGE,
-  HP_TRIG_SINGEDGECORNER1,
-  HP_TRIG,
-  HP_NONE,
-};
-int reftrig_singedges3_newels[][8] =
-{
-  //  { 1, 4, 9, 5 },
-  { 1, 4, 9 },
-  { 5, 1, 9 },
-  { 4, 2, 6, 9 },
-  { 7, 5, 9, 8 },
-  { 3, 7, 8 },
-  { 6, 8, 9 },
-};
-HPRef_Struct reftrig_singedges3 =
-{
-  HP_TRIG,
-  reftrig_singedges3_splitedges, 
-  reftrig_singedges3_splitfaces, 
-  0,
-  reftrig_singedges3_newelstypes, 
-  reftrig_singedges3_newels
-};
+	// check, if point has as least 3 different surfs:
+	ARRAY<INDEX_3, PointIndex::BASE> surfonpoint(mesh.GetNP());
 
+	for (int i = 1; i <= mesh.GetNP(); i++)
+	  surfonpoint.Elem(i) = INDEX_3(0,0,0);
+      
+	for (int i = 1; i <= mesh.GetNSeg(); i++)
+	  {
+	    const Segment & seg = mesh.LineSegment(i);
+	    int ind = seg.edgenr;
 
+	    if (seg.singedge_left)
+	      {
+		INDEX_2 i2 (mesh.LineSegment(i).p1, 
+			    mesh.LineSegment(i).p2);
+		edges.Set (i2, 1);
+		edgepoint.Set(i2.I1());
+		edgepoint.Set(i2.I2());
+	      
+		edgepoint_dom.Set (INDEX_2(mesh.LineSegment(i).domin, i2.I1()), 1);
+		edgepoint_dom.Set (INDEX_2(mesh.LineSegment(i).domin, i2.I2()), 1);
+	      }
+	    if (seg.singedge_right)
+	      {
+		INDEX_2 i2 (mesh.LineSegment(i).p2, 
+			    mesh.LineSegment(i).p1);
+		edges.Set (i2, 1);
+		edgepoint.Set(i2.I1());
+		edgepoint.Set(i2.I2());
+
+		edgepoint_dom.Set (INDEX_2(mesh.LineSegment(i).domout, i2.I1()), 1);
+		edgepoint_dom.Set (INDEX_2(mesh.LineSegment(i).domout, i2.I2()), 1);
+	      }
+	    // (*testout) << "seg = " << ind << ", " << seg.p1 << "-" << seg.p2 << endl;
 
 
+	    if (seg.singedge_left || seg.singedge_right)
+	      {
+		for (int j = 0; j < 2; j++)
+		  {
+		    int pi = (j == 0) ? seg.p1 : seg.p2;
+		    INDEX_3 & i3 = surfonpoint.Elem(pi);
+		    if (ind != i3.I1() &&
+			ind != i3.I2())
+		      {
+			i3.I1() = i3.I2();
+			i3.I2() = ind;
+		      }
+		  }
+	      }
+	  }
 
 
-// HP_TRIG_SINGEDGES23
-int reftrig_singedges23_splitedges[][3] =
-{
-  { 1, 2, 4 },
-  { 1, 3, 5 },
-  { 2, 1, 6 },
-  { 2, 3, 7 },
-  { 3, 1, 8 },
-  { 3, 2, 9 },
-  { 0, 0, 0 }
-};
-int reftrig_singedges23_splitfaces[][4] =
-{
-  { 1, 2, 3, 10 },
-  { 0, 0, 0, 0 }
-};
+	for (int i = 1; i <= mesh.GetNP(); i++)
+	  {
+	    // mark points for refinement that are in corners between two anisotropic edges 
+	    if (surfonpoint.Get(i).I1())
+	      {
+		cornerpoint.Set(i);
+		edgepoint.Set(i);
+	      }
+	
+	    // mark points for refinement that are explicity specified in input file
+	    if (mesh.Point(i).IsSingular())
+	      {
+		cornerpoint.Set(i);
+		edgepoint.Set(i);
+	      }
+	  }
 
-HPREF_ELEMENT_TYPE reftrig_singedges23_newelstypes[] =
-{
-  //  HP_QUAD_SINGEDGES,
-  HP_TRIG_SINGEDGECORNER1,
-  HP_TRIG_SINGEDGECORNER2,
-
-  HP_QUAD_SINGEDGE,
-  HP_QUAD_SINGEDGE,
-  HP_TRIG_SINGEDGECORNER2,
-  HP_TRIG_SINGEDGECORNER1,
-  HP_TRIG,
-  HP_NONE,
-};
-int reftrig_singedges23_newels[][8] =
-{
-  //  { 1, 4, 10, 5 },
-  { 1 , 4, 10 },
-  { 5, 1, 10 },
-  { 4, 6, 7, 10 },
-  { 8, 5, 10, 9 },
-  { 6, 2, 7 },
-  { 3, 8, 9 },
-  { 7, 9, 10 },
-};
-HPRef_Struct reftrig_singedges23 =
-{
-  HP_TRIG,
-  reftrig_singedges23_splitedges, 
-  reftrig_singedges23_splitfaces, 
-  0,
-  reftrig_singedges23_newelstypes, 
-  reftrig_singedges23_newels
-};
+	edgepoint.Or (cornerpoint);
 
+	(*testout) << "cornerpoints: " << endl << cornerpoint << endl
+		   << "edgepoints: " << endl << edgepoint << endl;
+      }
 
 
 
 
+    int cnt_undef = 0, cnt_nonimplement = 0;
+    ARRAY<int> misses(10000);
+    misses = 0;
 
+    for (ElementIndex i = 0; i < mesh.GetNE(); i++)
+      {
+	Element & el = mesh[i];
+      
+	HPREF_ELEMENT_TYPE type = HP_NONE;
+	int pnums[8] = { 0 };
+      
+      
+	switch (el.GetType())
+	  {
+	  case TET:
+	    {
+	      int ep1, ep2, ep3, ep4, cp1, cp2, cp3, cp4, fp1, fp2, fp3, fp4;
+	      int isedge1, isedge2, isedge3, isedge4, isedge5, isedge6;
+	      int isfedge1, isfedge2, isfedge3, isfedge4, isfedge5, isfedge6;
+	      int isface1, isface2, isface3, isface4;
 
+	      for (int j = 0; j < 4; j++)
+		for (int k = 0; k < 4; k++)
+		  {
+		    if (j == k) continue;
+		    if (type) break;
+	    
+		    int pi3 = 0;
+		    while (pi3 == j || pi3 == k) pi3++;
+		    int pi4 = 6 - j - k - pi3;
+
+		    // preserve orientation
+		    int sort[4];
+		    sort[0] = j; sort[1] = k; sort[2] = pi3; sort[3] = pi4;
+		    int cnt = 0;
+		    for (int jj = 0; jj < 4; jj++)
+		      for (int kk = 0; kk < 3; kk++)
+			if (sort[kk] > sort[kk+1])
+			  {
+			    cnt++;
+			    Swap (sort[kk], sort[kk+1]);
+			  }
+		    if (cnt % 2 == 1) Swap (pi3, pi4);
+
+		    ep1 = edgepoint.Test (el[j]);
+		    ep2 = edgepoint.Test (el[k]);
+		    ep3 = edgepoint.Test (el[pi3]);
+		    ep4 = edgepoint.Test (el[pi4]);
+
+		    cp1 = cornerpoint.Test (el[j]);
+		    cp2 = cornerpoint.Test (el[k]);
+		    cp3 = cornerpoint.Test (el[pi3]);
+		    cp4 = cornerpoint.Test (el[pi4]);
+	    
+		    isedge1 = edges.Used (INDEX_2::Sort (el[j], el[k]));
+		    isedge2 = edges.Used (INDEX_2::Sort (el[j], el[pi3]));
+		    isedge3 = edges.Used (INDEX_2::Sort (el[j], el[pi4]));
+		    isedge4 = edges.Used (INDEX_2::Sort (el[k], el[pi3]));
+		    isedge5 = edges.Used (INDEX_2::Sort (el[k], el[pi4]));
+		    isedge6 = edges.Used (INDEX_2::Sort (el[pi3], el[pi4]));
+
+		    isface1 = isface2 = isface3 = isface4 = 0;
+		    for (int l = 0; l < 4; l++)
+		      {
+			INDEX_3 i3;
+			switch (l)
+			  {
+			  case 0: i3 = INDEX_3 (el[k], el[pi3], el[pi4]); break;
+			  case 1: i3 = INDEX_3 (el[j], el[pi3], el[pi4]); break;
+			  case 2: i3 = INDEX_3 (el[j], el[k], el[pi4]); break;
+			  case 3: i3 = INDEX_3 (el[j], el[k], el[pi3]); break;
+			  }
+			i3.Sort();
+			if (faces.Used (i3))
+			  {
+			    int domnr = faces.Get(i3);
+			    if (domnr == -1 || domnr == el.GetIndex())
+			      {
+				switch (l)
+				  {
+				  case 0: isface1 = 1; break;
+				  case 1: isface2 = 1; break;
+				  case 2: isface3 = 1; break;
+				  case 3: isface4 = 1; break;
+				  }
+			      }
+			  }
+		      }
+		    /*
+		    isface1 = faces.Used (INDEX_3::Sort (el[k], el[pi3], el[pi4]));
+		    isface2 = faces.Used (INDEX_3::Sort (el[j], el[pi3], el[pi4]));
+		    isface3 = faces.Used (INDEX_3::Sort (el[j], el[k], el[pi4]));
+		    isface4 = faces.Used (INDEX_3::Sort (el[j], el[k], el[pi3]));
+		    */
+
+		    isfedge1 = isfedge2 = isfedge3 = isfedge4 = isfedge5 = isfedge6 = 0;
+		    for (int l = 0; l < 6; l++)
+		      {
+			INDEX_2 i2;
+			switch (l)
+			  {
+			  case 0: i2 = INDEX_2 (el[j], el[k]); break;
+			  case 1: i2 = INDEX_2 (el[j], el[pi3]); break;
+			  case 2: i2 = INDEX_2 (el[j], el[pi4]); break;
+			  case 3: i2 = INDEX_2 (el[k], el[pi3]); break;
+			  case 4: i2 = INDEX_2 (el[k], el[pi4]); break;
+			  case 5: i2 = INDEX_2 (el[pi3], el[pi4]); break;
+			  }
+			i2.Sort();
+			if (face_edges.Used (i2))
+			  {
+			    int domnr = face_edges.Get(i2);
+			    if (domnr == -1 || domnr == el.GetIndex())
+			      {
+				switch (l)
+				  {
+				  case 0: isfedge1 = 1; break;
+				  case 1: isfedge2 = 1; break;
+				  case 2: isfedge3 = 1; break;
+				  case 3: isfedge4 = 1; break;
+				  case 4: isfedge5 = 1; break;
+				  case 5: isfedge6 = 1; break;
+				  }
+			      }
+			  }
+		      }
+		    /*
+		    isfedge1 = face_edges.Used (INDEX_2::Sort (el[j], el[k]));
+		    isfedge2 = face_edges.Used (INDEX_2::Sort (el[j], el[pi3]));
+		    isfedge3 = face_edges.Used (INDEX_2::Sort (el[j], el[pi4]));
+		    isfedge4 = face_edges.Used (INDEX_2::Sort (el[k], el[pi3]));
+		    isfedge5 = face_edges.Used (INDEX_2::Sort (el[k], el[pi4]));
+		    isfedge6 = face_edges.Used (INDEX_2::Sort (el[pi3], el[pi4]));
+		    */
+
+		    fp1 = fp2 = fp3 = fp4 = 0;
+		    for (int l = 0; l < 4; l++)
+		      {
+			int pi;
+			switch (l)
+			  {
+			  case 0: pi = el[j]; break;
+			  case 1: pi = el[k]; break;
+			  case 2: pi = el[pi3]; break;
+			  case 3: pi = el[pi4]; break;
+			  }
+			int domnr = facepoint[pi];
+			if (domnr == -1 || domnr == el.GetIndex())
+			  {
+			    switch (l)
+			      {
+			      case 0: fp1 = 1; break;
+			      case 1: fp2 = 1; break;
+			      case 2: fp3 = 1; break;
+			      case 3: fp4 = 1; break;
+			      }
+			  }
+		      }
 
+		    /*
+		    fp1 = facepoint[el[j]] != 0;
+		    fp2 = facepoint[el[k]] != 0;
+		    fp3 = facepoint[el[pi3]] != 0;
+		    fp4 = facepoint[el[pi4]] != 0;
+		    */
 
 
+		    switch (isface1+isface2+isface3+isface4)
+		      {
+		      case 0:
+			{
+			  isedge1 |= isfedge1;
+			  isedge2 |= isfedge2;
+			  isedge3 |= isfedge3;
+			  isedge4 |= isfedge4;
+			  isedge5 |= isfedge5;
+			  isedge6 |= isfedge6;
+
+			  ep1 |= fp1;
+			  ep2 |= fp2;
+			  ep3 |= fp3;
+			  ep4 |= fp4;
+
+			  switch (isedge1+isedge2+isedge3+isedge4+isedge5+isedge6)
+			    {
+			    case 0:
+			      {		
+				if (!ep1 && !ep2 && !ep3 && !ep4)
+				  type = HP_TET;
+				
+				if (ep1 && !ep2 && !ep3 && !ep4)
+				  type = HP_TET_0E_1V;
+
+				if (ep1 && ep2 && !ep3 && !ep4)
+				  type = HP_TET_0E_2V;
+
+				if (ep1 && ep2 && ep3 && !ep4)
+				  type = HP_TET_0E_3V;
+
+				if (ep1 && ep2 && ep3 && ep4)
+				  type = HP_TET_0E_4V;
+
+				break;
+			      }
+		
+			    case 1:
+			      {
+				if (!isedge1) break;
+		  
+				if (!cp1 && !cp2 && !ep3 && !ep4)
+				  type = HP_TET_1E_0V;
+		    
+				if (cp1 && !cp2 && !ep3 && !ep4)
+				  type = HP_TET_1E_1VA;
 
+				if (!cp1 && !cp2 && !ep3 && ep4)
+				  type = HP_TET_1E_1VB;
 
+				if (cp1 && cp2 && !ep3 && !ep4)
+				  type = HP_TET_1E_2VA;
 
-// HP_QUAD
-int refquad_splitedges[][3] =
-{
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE refquad_newelstypes[] =
-{
-  HP_QUAD,
-  HP_NONE,
-};
-int refquad_newels[][8] =
-{
-  { 1, 2, 3, 4 },
-};
-HPRef_Struct refquad =
-{
-  HP_QUAD,
-  refquad_splitedges, 
-  0, 0,
-  refquad_newelstypes, 
-  refquad_newels
-};
+				if (cp1 && !cp2 && ep3 && !ep4)
+				  type = HP_TET_1E_2VB;
 
+				if (cp1 && !cp2 && !ep3 && ep4)
+				  type = HP_TET_1E_2VC;
 
+				if (!cp1 && !cp2 && ep3 && ep4)
+				  type = HP_TET_1E_2VD;
 
+				if (cp1 && cp2 && ep3 && !ep4)
+				  type = HP_TET_1E_3VA;
 
+				if (cp1 && !cp2 && ep3 && ep4)
+				  type = HP_TET_1E_3VB;
 
+				if (cp1 && cp2 && ep3 && ep4)
+				  type = HP_TET_1E_4V;
+		  
+				break;
+			      }
+			    case 2:
+			      {
+				if (isedge1 && isedge2)
+				  {
+				    if (!cp2 && !cp3 && !ep4)
+				      type = HP_TET_2EA_0V;
+
+				    if (cp2 && !cp3 && !ep4)
+				      type = HP_TET_2EA_1VA;
+				    if (!cp2 && cp3 && !ep4)
+				      type = HP_TET_2EA_1VB;
+
+				    if (!cp2 && !cp3 && ep4)
+				      type = HP_TET_2EA_1VC;
+
+				    if (cp2 && cp3 && !ep4)
+				      type = HP_TET_2EA_2VA;
+				    if (cp2 && !cp3 && ep4)
+				      type = HP_TET_2EA_2VB;
+				    if (!cp2 && cp3 && ep4)
+				      type = HP_TET_2EA_2VC;
+
+				    if (cp2 && cp3 && ep4)
+				      type = HP_TET_2EA_3V;
+				  }
+				if (isedge1 && isedge6)
+				  {
+				    if (!cp1 && !cp2 && !cp3 && !cp4)
+				      type = HP_TET_2EB_0V;
+				    if (cp1 && !cp2 && !cp3 && !cp4)
+				      type = HP_TET_2EB_1V;
+				    if (cp1 && cp2 && !cp3 && !cp4)
+				      type = HP_TET_2EB_2VA;
+				    if (cp1 && !cp2 && cp3 && !cp4)
+				      type = HP_TET_2EB_2VB;
+				    if (cp1 && !cp2 && !cp3 && cp4)
+				      type = HP_TET_2EB_2VC;
+				    if (cp1 && cp2 && cp3 && !cp4)
+				      type = HP_TET_2EB_3V;
+				    if (cp1 && cp2 && cp3 && cp4)
+				      type = HP_TET_2EB_4V;
+				  }
+			      }
+			    case 3:
+			      {
+				if (isedge1 && isedge2 && isedge3)
+				  {
+				    if (!cp2 && !cp3 && !cp4)
+				      type = HP_TET_3EA_0V;
+				    if (cp2 && !cp3 && !cp4)
+				      type = HP_TET_3EA_1V;
+				    if (cp2 && cp3 && !cp4)
+				      type = HP_TET_3EA_2V;
+				    if (cp2 && cp3 && cp4)
+				      type = HP_TET_3EA_3V;
+				  }
+				if (isedge1 && isedge3 && isedge4)
+				  {
+				    if (!cp3 && !cp4)
+				      type = HP_TET_3EB_0V;
+				    if (cp3 && !cp4)
+				      type = HP_TET_3EB_1V;
+				    if (cp3 && cp4)
+				      type = HP_TET_3EB_2V;
+				  }
+				if (isedge1 && isedge2 && isedge5)
+				  {
+				    if (!cp3 && !cp4)
+				      type = HP_TET_3EC_0V;
+				    if (cp3 && !cp4)
+				      type = HP_TET_3EC_1V;
+				    if (cp3 && cp4)
+				      type = HP_TET_3EC_2V;
+				  }
+				break;
+			      }
+			    }
+			  break;
+			}
+
+
+
+		      case 1:  // one singular face
+			{
+			  if (!isface1) break;
+
+			  switch (isfedge1+isfedge2+isfedge3+isedge4+isedge5+isedge6)
+			    {
+			    case 0:
+			      {
+				if (!fp1 && !ep2 && !ep3 && !ep4)
+				  type = HP_TET_1F_0E_0V;
+				if (fp1 && !ep2 && !ep3 && !ep4)
+				  type = HP_TET_1F_0E_1VB;
+				if (!fp1 && ep2 && !ep3 & !ep4)
+				  type = HP_TET_1F_0E_1VA;
+				break;
+			      }
+			    case 1:
+			      {
+				if (isfedge1)
+				  {
+				    if (!ep1 && !ep3 && !ep4)
+				      type = HP_TET_1F_1EA_0V;
+				  }
+				if (isedge4) // V1-V3
+				  {
+				    if (!ep1 && !cp2 && !cp3 && !ep4)
+				      type = HP_TET_1F_1EB_0V;
+				  }
+				break;
+			      }
+			    }
+			  break;
+			}
+
+
+		      case 2:  // one singular face
+			{
+			  if (!isface1 || !isface2) break;
+
+			  switch (isfedge1+isedge2+isedge3+isedge4+isedge5)
+			    {
+			    case 0:
+			      {
+				if (!ep1 && !ep2 && !cp3 && !cp4)
+				  type = HP_TET_2F_0E_0V;
+				break;
+			      }
+			    }
+			  break;
+			}
 
 
-// HP_QUAD_SINGCORNER
-int refquad_singcorner_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 4, 6 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE refquad_singcorner_newelstypes[] =
-{
-  HP_TRIG_SINGCORNER,
-  HP_QUAD,
-  HP_TRIG,
-  HP_NONE,
-};
-int refquad_singcorner_newels[][8] =
-{
-  { 1, 5, 6 },
-  { 2, 4, 6, 5 },
-  { 2, 3, 4 },
-};
-HPRef_Struct refquad_singcorner =
-{
-  HP_QUAD,
-  refquad_singcorner_splitedges, 
-  0, 0,
-  refquad_singcorner_newelstypes, 
-  refquad_singcorner_newels
-};
+		      }
 
+		    if (type != HP_NONE)
+		      {
+			pnums[0] = el[j];
+			pnums[1] = el[k];
+			pnums[2] = el[pi3];
+			pnums[3] = el[pi4];
+			break;
+		      }
+		  }
 
+	      /*
+		if (type != HP_TET_2EB_2VA)
+		type = HP_NONE;
+	      */
 
+	      if (type == HP_NONE)
+		{
+		  cnt_undef++;
+		  (*testout) << "undefined element" << endl
+			     << "cp = " << cp1 << cp2 << cp3 << cp4 << endl
+			     << "ep = " << ep1 << ep2 << ep3 << ep4 << endl
+			     << "isedge = " << isedge1 << isedge2 << isedge3 
+			     << isedge4 << isedge5 << isedge6 << endl;
+		  cout << "undefined element !!! " << endl;
+		}
 
+	      break;
+	    }
 
-// HP_DUMMY_QUAD_SINGCORNER
-int refdummyquad_singcorner_splitedges[][3] =
-{
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE refdummyquad_singcorner_newelstypes[] =
-{
-  HP_TRIG_SINGCORNER,
-  HP_TRIG,
-  HP_NONE,
-};
-int refdummyquad_singcorner_newels[][8] =
-{
-  { 1, 2, 4 },
-  { 4, 2, 3 },
-};
-HPRef_Struct refdummyquad_singcorner =
-{
-  HP_QUAD,
-  refdummyquad_singcorner_splitedges, 
-  0, 0,
-  refdummyquad_singcorner_newelstypes, 
-  refdummyquad_singcorner_newels
-};
+	  case PRISM:
+	    {
+	      int pi1, pi2, pi3, pi4, pi5, pi6;
+	      int ep1, ep2, ep3, ep4, ep5, ep6, cp1, cp2, cp3, cp4, cp5, cp6;
 
+	      int ishedge1, ishedge2, ishedge3, ishedge4, ishedge5, ishedge6;
+	      int isvedge1, isvedge2, isvedge3;
+	    
+	      for (int j = 1; j <= 3; j++)
+		{
+		  if (type) break;
+
+		  pi1 = j;
+		  pi2 = pi1%3 + 1;
+		  pi3 = pi2%3 + 1;
+		  pi4 = pi1+3;
+		  pi5 = pi2+3;
+		  pi6 = pi3+3;
+
+		  ep1 = edgepoint.Test (el.PNum (pi1));
+		  ep2 = edgepoint.Test (el.PNum (pi2));
+		  ep3 = edgepoint.Test (el.PNum (pi3));
+		  ep4 = edgepoint.Test (el.PNum (pi4));
+		  ep5 = edgepoint.Test (el.PNum (pi5));
+		  ep6 = edgepoint.Test (el.PNum (pi6));
+
+		  cp1 = cornerpoint.Test (el.PNum (pi1));
+		  cp2 = cornerpoint.Test (el.PNum (pi2));
+		  cp3 = cornerpoint.Test (el.PNum (pi3));
+		  cp4 = cornerpoint.Test (el.PNum (pi4));
+		  cp5 = cornerpoint.Test (el.PNum (pi5));
+		  cp6 = cornerpoint.Test (el.PNum (pi6));
+	    
+		  INDEX_2 i2 = INDEX_2::Sort(el.PNum (pi1), el.PNum (pi4));
+		  isvedge1 = edges.Used (i2) || ep1 || ep4;
 
+		  i2 = INDEX_2::Sort(el.PNum (pi2), el.PNum (pi5));
+		  isvedge2 = edges.Used (i2) || ep2 || ep5;
 
+		  i2 = INDEX_2::Sort(el.PNum (pi3), el.PNum (pi6));
+		  isvedge3 = edges.Used (i2) || ep3 || ep6;
 
 
+		  ishedge1 = edges.Used (INDEX_2::Sort(el.PNum (pi1), el.PNum (pi2)));
+		  ishedge2 = edges.Used (INDEX_2::Sort(el.PNum (pi2), el.PNum (pi3)));
+		  ishedge3 = edges.Used (INDEX_2::Sort(el.PNum (pi3), el.PNum (pi1)));
 
 
-// HP_QUAD_SINGEDGE
-int refquad_singedge_splitedges[][3] =
-{
-  { 1, 4, 5 },
-  { 2, 3, 6 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE refquad_singedge_newelstypes[] =
-{
-  HP_QUAD_SINGEDGE,
-  HP_QUAD,
-  HP_NONE,
-};
-int refquad_singedge_newels[][8] =
-{
-  { 1, 2, 6, 5 },
-  { 5, 6, 3, 4 },
-};
-HPRef_Struct refquad_singedge =
-{
-  HP_QUAD,
-  refquad_singedge_splitedges, 
-  0, 0,
-  refquad_singedge_newelstypes, 
-  refquad_singedge_newels
-};
 
+		  switch (ishedge1 + ishedge2 + ishedge3)
+		    {
+		    case 0:
+		      {
+			if (!isvedge1 && !isvedge2 && !isvedge3)
+			  type = HP_PRISM;
+			else if (isvedge1 && !isvedge2 && !isvedge3)
+			  type = HP_PRISM_SINGEDGE;
+			else if (isvedge1 && isvedge2 && !isvedge3)
+			  type = HP_PRISM_SINGEDGE_V12;
+			break;
+		      }
+		    case 1:
+		      {
+			if (ishedge1)
+			  type = HP_PRISM_SINGEDGE_H1;
+			break;
+		      }
+		    case 2:
+		      {
+			if (ishedge1 && ishedge2)
+			  type = HP_PRISM_SINGEDGE_H12;
+			break;
+		      }
+		    }
 
+		  if (type != HP_NONE)
+		    {
+		      cout << "classified element, type = " << type << ", el = "  << el << endl;
+		      pnums[0] = el.PNum (pi1);
+		      pnums[1] = el.PNum (pi2);
+		      pnums[2] = el.PNum (pi3);
+		      pnums[3] = el.PNum (pi4);
+		      pnums[4] = el.PNum (pi5);
+		      pnums[5] = el.PNum (pi6);
+		      break;
+		    }
+		}
+	      break;
+	    }
+	  default:
+	    {
+	      cerr << "hp-refinement not defined for element" << endl;
+	    }
+	  }
 
 
+	if (type == HP_NONE)
+	  {
+	    cout << "element is HP_NONE: " << el << endl;
+	  }
 
-// HP_QUAD_SINGEDGES
-int refquad_singedges_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 4, 6 },
-  { 2, 3, 7 },
-  { 4, 3, 8 },
-  { 0, 0, 0 }
-};
-int refquad_singedges_splitfaces[][4] =
-{
-  { 1, 2, 4, 9 },
-  { 0, 0, 0, 0 },
-};
-HPREF_ELEMENT_TYPE refquad_singedges_newelstypes[] =
-{
-  HP_TRIG_SINGEDGECORNER1,
-  HP_TRIG_SINGEDGECORNER2,
-  HP_QUAD_SINGEDGE,
-  HP_QUAD_SINGEDGE,
-  HP_QUAD,
-  HP_NONE,
-};
-int refquad_singedges_newels[][8] =
-{
-  { 1, 5, 9 },
-  { 6, 1, 9 },
-  { 5, 2, 7, 9 },
-  { 4, 6, 9, 8 },
-  { 9, 7, 3, 8 },
-};
-HPRef_Struct refquad_singedges =
-{
-  HP_QUAD,
-  refquad_singedges_splitedges, 
-  refquad_singedges_splitfaces, 
-  0,
-  refquad_singedges_newelstypes, 
-  refquad_singedges_newels
-};
-
-
-
-
-
-
-
-
-
-
-
-
-
-// HP_TET
-int reftet_splitedges[][3] =
-{
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftet_newelstypes[] =
-{
-  HP_TET,
-  HP_NONE,
-};
-int reftet_newels[][8] =
-{
-  { 1, 2, 3, 4 },
-};
-HPRef_Struct reftet =
-{
-  HP_TET,
-  reftet_splitedges, 
-  0, 0,
-  reftet_newelstypes, 
-  reftet_newels
-};
-
-
-
-/* *********** Tet - Refinement - 0 edges *************** */
-
-// HP_TET_0E_1V
-int reftet_0e_1v_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftet_0e_1v_newelstypes[] =
-{
-  HP_TET_0E_1V,
-  HP_PRISM,
-  HP_NONE,
-};
-int reftet_0e_1v_newels[][8] =
-{
-  { 1, 5, 6, 7 },
-  { 5, 6, 7, 2, 3, 4 }
-};
-HPRef_Struct reftet_0e_1v =
-{
-  HP_TET,
-  reftet_0e_1v_splitedges, 
-  0, 0,
-  reftet_0e_1v_newelstypes, 
-  reftet_0e_1v_newels
-};
-
+	if (!Get_HPRef_Struct (type)) 
+	  {
+	    (*testout) << "case " << type << " not implemented " << endl;
+	    cnt_nonimplement++;
+	    misses[type]++;
+	  }
+      
+	HPRefElement hpel;
+	hpel.type = type;
+	for (int j = 0; j < 8; j++)
+	  hpel.pnums[j] = pnums[j];
+	hpel.index = el.GetIndex();
+	hpel.level = 1;
+	hpel.coarse_elnr = i;
+
+	static const double points[4][3] = 
+	  { { 1, 0, 0 }, { 0, 1, 0 }, { 0, 0, 1 }, { 0, 0, 0 } };
+	for (int j = 0; j < 4; j++)
+	  for (int k = 0; k < 4; k++)
+	    if (pnums[j] == el[k])
+	      {
+		hpel.param[j][0] = points[k][0];
+		hpel.param[j][1] = points[k][1];
+		hpel.param[j][2] = points[k][2];
+	      }
 
+	elements.Append (hpel);
+      }
 
-// HP_TET_0E_2V
-int reftet_0e_2v_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 1, 8 },
-  { 2, 3, 9 },
-  { 2, 4, 10 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftet_0e_2v_newelstypes[] =
-{
-  HP_TET_0E_1V,
-  HP_TET_0E_1V,
-  HP_PRISM,
-  HP_PRISM,
-  HP_NONE,
-};
-int reftet_0e_2v_newels[][8] =
-{
-  { 1, 5, 6, 7 },
-  { 2, 10, 9, 8 },
-  { 5, 6, 7, 8, 9, 10 },
-  { 4, 10, 7, 3, 9, 6 },
-};
-HPRef_Struct reftet_0e_2v =
-{
-  HP_TET,
-  reftet_0e_2v_splitedges, 
-  0, 0,
-  reftet_0e_2v_newelstypes, 
-  reftet_0e_2v_newels
-};
+    cout << "undefined elements: " << cnt_undef << endl;
+    cout << "non-implemented: " << cnt_nonimplement << endl;
 
+    for (int i = 0; i < misses.Size(); i++)
+      if (misses[i])
+	cout << "missing case " << i << " occured " << misses[i] << " times" << endl;
 
+    for (int i = 1; i <= mesh.GetNSE(); i++)
+      {
+	int j, k, pi3, pi4;
+	Element2d & el = mesh.SurfaceElement(i);
+      
+	HPREF_ELEMENT_TYPE type = HP_NONE;
+	int pnums[8] = { 0 };
 
+	switch (el.GetType())
+	  {
+	  case TRIG:
+	    {
+	      for (j = 1; j <= 3; j++)
+		{
+		  int ep1 = edgepoint.Test (el.PNumMod (j));
+		  int ep2 = edgepoint.Test (el.PNumMod (j+1));
+		  int ep3 = edgepoint.Test (el.PNumMod (j+2));
+		
+		  if (mesh.GetDimension() == 2)
+		    {
+		      ep1 = edgepoint_dom.Used (INDEX_2 (mesh.SurfaceElement(i).GetIndex(), el.PNumMod(j)));
+		      ep2 = edgepoint_dom.Used (INDEX_2 (mesh.SurfaceElement(i).GetIndex(), el.PNumMod(j+1)));
+		      ep3 = edgepoint_dom.Used (INDEX_2 (mesh.SurfaceElement(i).GetIndex(), el.PNumMod(j+2)));
+		    }
 
+		  int cp1 = cornerpoint.Test (el.PNumMod (j));
+		  int cp2 = cornerpoint.Test (el.PNumMod (j+1));
+		  int cp3 = cornerpoint.Test (el.PNumMod (j+2));
 
-// HP_TET_0E_3V
-int reftet_0e_3v_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 1, 8 },
-  { 2, 3, 9 },
-  { 2, 4, 10 },
-  { 3, 1, 11 },
-  { 3, 2, 12 },
-  { 3, 4, 13 },
-  { 0, 0, 0 }
-};
-int reftet_0e_3v_splitfaces[][4] =
-  {
-    { 1, 2, 3, 14 },
-    { 2, 3, 1, 15 },
-    { 3, 1, 2, 16 },
-    { 0, 0, 0, 0 },
-  };
-HPREF_ELEMENT_TYPE reftet_0e_3v_newelstypes[] =
-{
-  HP_PYRAMID_0E_1V,
-  HP_PYRAMID_0E_1V,
-  HP_PYRAMID_0E_1V,
-  HP_PRISM,
-  HP_PRISM,
-  HP_PRISM,
-  HP_PRISM,
-  HP_TET,
-  HP_NONE,
-};
-int reftet_0e_3v_newels[][8] =
-{
-  { 1, 5, 14, 6, 7 },
-  { 2, 9, 15, 8, 10 },
-  { 3, 11, 16, 12, 13 },
-  { 5, 14, 7, 8, 15, 10 },
-  { 9, 15, 10, 12, 16, 13 },
-  { 6, 7, 14, 11, 13, 16 },
-  { 14, 15, 16, 7, 10, 13 },
-  { 7, 10, 13, 4 }
-};
-HPRef_Struct reftet_0e_3v =
-{
-  HP_TET,
-  reftet_0e_3v_splitedges, 
-  reftet_0e_3v_splitfaces, 
-  0,
-  reftet_0e_3v_newelstypes, 
-  reftet_0e_3v_newels
-};
+		  ep1 |= cp1;
+		  ep2 |= cp2;
+		  ep3 |= cp3;
+		
+		  INDEX_2 i2;
+		  i2 = INDEX_2(el.PNumMod (j), el.PNumMod (j+1));
+		  // i2.Sort();
+		  int isedge1 = edges.Used (i2);
+		  i2 = INDEX_2(el.PNumMod (j+1), el.PNumMod (j+2));
+		  // i2.Sort();
+		  int isedge2 = edges.Used (i2);
+		  i2 = INDEX_2(el.PNumMod (j+2), el.PNumMod (j+3));
+		  // i2.Sort();
+		  int isedge3 = edges.Used (i2);
+		
+		  if (isedge1 + isedge2 + isedge3 == 0)
+		    {
+		      if (!ep1 && !ep2 && !ep3)
+			type = HP_TRIG;
+		    
+		      if (ep1 && !ep2 && !ep3)
+			type = HP_TRIG_SINGCORNER;
+		    
+		      if (ep1 && ep2 && !ep3)
+			type = HP_TRIG_SINGCORNER12;
+		    
+		      if (ep1 && ep2 && ep3)
+			{
+			  if (mesh.GetDimension() == 2)
+			    type = HP_TRIG_SINGCORNER123_2D;
+			  else
+			    type = HP_TRIG_SINGCORNER123;
+			}
+		    
+		      if (type != HP_NONE)
+			{
+			  pnums[0] = el.PNumMod (j);
+			  pnums[1] = el.PNumMod (j+1);
+			  pnums[2] = el.PNumMod (j+2);
+			  break;
+			}
+		    }
+		
+		  if (isedge1 && !isedge2 && !isedge3)
+		    {
+		      int code = 0;
+		      if (cp1) code += 1;
+		      if (cp2) code += 2;
+		      if (ep3) code += 4;
+		    
+		      HPREF_ELEMENT_TYPE types[] =
+			{
+			  HP_TRIG_SINGEDGE, 
+			  HP_TRIG_SINGEDGECORNER1, 
+			  HP_TRIG_SINGEDGECORNER2,
+			  HP_TRIG_SINGEDGECORNER12, 
+			  HP_TRIG_SINGEDGECORNER3, 
+			  HP_TRIG_SINGEDGECORNER13, 
+			  HP_TRIG_SINGEDGECORNER23, 
+			  HP_TRIG_SINGEDGECORNER123, 
+			};
+		      type = types[code];
+		      pnums[0] = el.PNumMod (j);
+		      pnums[1] = el.PNumMod (j+1);
+		      pnums[2] = el.PNumMod (j+2);
+		      break;
+		    }
+		
+		
+		  if (isedge1 && !isedge2 && isedge3)
+		    {
+		      if (!cp3)
+			{
+			  if (!cp2) type = HP_TRIG_SINGEDGES;
+			  else      type = HP_TRIG_SINGEDGES2;
+			}
+		      else
+			{
+			  if (!cp2) type = HP_TRIG_SINGEDGES3;
+			  else      type = HP_TRIG_SINGEDGES23;
+			}
+		    
+		      pnums[0] = el.PNumMod (j);
+		      pnums[1] = el.PNumMod (j+1);
+		      pnums[2] = el.PNumMod (j+2);
+		      break;
+		    }
+		
+		  if (isedge1 && isedge2 && isedge3)
+		    {
+		      type = HP_TRIG_3SINGEDGES;
+		      pnums[0] = el.PNumMod (j);
+		      pnums[1] = el.PNumMod (j+1);
+		      pnums[2] = el.PNumMod (j+2);
+		      break;
+		    }
+		}
+	      break;
+	    }
+	  case QUAD:
+	    {
+	      int ep1, ep2, ep3, ep4, cp1, cp2, cp3, cp4;
+	      int isedge1, isedge2, isedge3, isedge4;
 
+	      for (j = 1; j <= 4; j++)
+		{
+		  ep1 = edgepoint.Test (el.PNumMod (j));
+		  ep2 = edgepoint.Test (el.PNumMod (j+1));
+		  ep3 = edgepoint.Test (el.PNumMod (j+2));
+		  ep4 = edgepoint.Test (el.PNumMod (j+3));
 
+		  if (mesh.GetDimension() == 2)
+		    {
+		      ep1 = edgepoint_dom.Used (INDEX_2 (mesh.SurfaceElement(i).GetIndex(), el.PNumMod(j)));
+		      ep2 = edgepoint_dom.Used (INDEX_2 (mesh.SurfaceElement(i).GetIndex(), el.PNumMod(j+1)));
+		      ep3 = edgepoint_dom.Used (INDEX_2 (mesh.SurfaceElement(i).GetIndex(), el.PNumMod(j+2)));
+		      ep4 = edgepoint_dom.Used (INDEX_2 (mesh.SurfaceElement(i).GetIndex(), el.PNumMod(j+3)));
+		    }
 
+		  cp1 = cornerpoint.Test (el.PNumMod (j));
+		  cp2 = cornerpoint.Test (el.PNumMod (j+1));
+		  cp3 = cornerpoint.Test (el.PNumMod (j+2));
+		  cp4 = cornerpoint.Test (el.PNumMod (j+3));
 
+		  ep1 |= cp1;
+		  ep2 |= cp2;
+		  ep3 |= cp3;
+		  ep4 |= cp4;
+		
 
-// HP_TET_0E_4V
-int reftet_0e_4v_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 1, 8 },
-  { 2, 3, 9 },
-  { 2, 4, 10 },
-  { 3, 1, 11 },
-  { 3, 2, 12 },
-  { 3, 4, 13 },
-  { 4, 1, 14 },
-  { 4, 2, 15 },
-  { 4, 3, 16 },
-  { 0, 0, 0 }
-};
-int reftet_0e_4v_splitfaces[][4] =
-  {
-    { 1, 2, 3, 17 },
-    { 1, 2, 4, 18 },
-    { 1, 3, 4, 19 },
-
-    { 2, 1, 3, 20 },
-    { 2, 1, 4, 21 },
-    { 2, 3, 4, 22 },
-
-    { 3, 1, 2, 23 },
-    { 3, 1, 4, 24 },
-    { 3, 2, 4, 25 },
-
-    { 4, 1, 2, 26 },
-    { 4, 1, 3, 27 },
-    { 4, 2, 3, 28 },
-    { 0, 0, 0, 0 },
-  };
-int reftet_0e_4v_splitelements[][5] =
-  {
-    { 1, 2, 3, 4, 29 },
-    { 2, 3, 4, 1, 30 },
-    { 3, 4, 1, 2, 31 },
-    { 4, 1, 2, 3, 32 },
-    { 0 },
-  };
-HPREF_ELEMENT_TYPE reftet_0e_4v_newelstypes[] =
-{
-  HP_HEX_0E_1V,
-  HP_HEX_0E_1V,
-  HP_HEX_0E_1V,
-  HP_HEX_0E_1V,
-  HP_PRISM, HP_PRISM, 
-  HP_PRISM, HP_PRISM, 
-  HP_PRISM, HP_PRISM, 
-  HP_PRISM, HP_PRISM, 
-  HP_PRISM, HP_PRISM, 
-  HP_PRISM, HP_PRISM, 
-  /*
-  HP_HEX, 
-  HP_HEX, 
-  HP_HEX, 
-  HP_HEX, 
-  HP_HEX, 
-  HP_HEX, 
-  */
-  HP_PRISM,
-  HP_PRISM,
-  HP_PRISM,
-  HP_PRISM,
-  HP_TET,
-  HP_NONE,
-};
-int reftet_0e_4v_newels[][8] =
-{
-  { 1, 5, 17, 6, 7, 18, 29, 19 },
-  { 2, 9, 20, 8, 10, 22, 30, 21 },
-  { 3, 11, 23, 12, 13, 24, 31, 25 },
-  { 4, 15, 26, 14, 16, 28, 32, 27 },
-  { 5, 17, 18, 8, 20, 21 },
-  { 18, 17, 29, 21, 20, 30 },
-  { 6, 19, 17,  11, 24, 23 },
-  { 17, 19, 29,  23, 24, 31 },
-  { 7, 18, 19, 14, 26, 27 },
-  { 19, 18, 29, 27, 26, 32 },
-  { 9, 20, 22, 12, 23, 25 },
-  { 22, 20, 30, 25, 23, 31 },
-  { 10, 22, 21, 15, 28, 26 },
-  { 21, 22, 30, 26, 28, 32 },
-  { 13, 24, 25, 16, 27, 28 },
-  { 25, 24, 31, 28, 27, 32 },
-  /*
-  { 5, 17, 29, 18, 8, 20, 30, 21 },
-  { 6, 19, 29, 17, 11, 24, 31, 23 },
-  { 7, 18, 29, 19, 14, 26, 32, 27 },
-  { 9, 20, 30, 22, 12, 23, 31, 25 },
-  { 10, 22, 30, 21, 15, 28, 32, 26 },
-  { 13, 24, 31, 25, 16, 27, 32, 28 },
-  */
-  { 17, 20, 23, 29, 30, 31 },
-  { 18, 26, 21, 29, 32, 30 },
-  { 19, 24, 27, 29, 31, 32 },
-  { 22, 28, 25, 30, 32, 31 },
-
-  { 29, 30, 31, 32 },
-};
-HPRef_Struct reftet_0e_4v =
-{
-  HP_TET,
-  reftet_0e_4v_splitedges, 
-  reftet_0e_4v_splitfaces, 
-  reftet_0e_4v_splitelements, 
-  reftet_0e_4v_newelstypes, 
-  reftet_0e_4v_newels
-};
+		  INDEX_2 i2;
+		  i2 = INDEX_2(el.PNumMod (j), el.PNumMod (j+1));
+		  // i2.Sort();
+		  isedge1 = edges.Used (i2);
+		  i2 = INDEX_2(el.PNumMod (j+1), el.PNumMod (j+2));
+		  // i2.Sort();
+		  isedge2 = edges.Used (i2);
+		  i2 = INDEX_2(el.PNumMod (j+2), el.PNumMod (j+3));
+		  // i2.Sort();
+		  isedge3 = edges.Used (i2);
+		  i2 = INDEX_2(el.PNumMod (j+3), el.PNumMod (j+4));
+		  // i2.Sort();
+		  isedge4 = edges.Used (i2);
+		
+		  int sumcp = cp1 + cp2 + cp3 + cp4;
+		  int sumep = ep1 + ep2 + ep3 + ep4;
+		  int sumedge = isedge1 + isedge2 + isedge3 + isedge4;
 
+		  switch (sumedge)
+		    {
+		    case 0:
+		      {
+			switch (sumep)
+			  {
+			  case 0: 
+			    type = HP_QUAD; 
+			    break;
+			  case 1: 
+			    if (ep1) type = HP_QUAD_SINGCORNER;
+			    break; 
+			  case 2:
+			    {
+			      if (ep1 && ep2) type = HP_QUAD_0E_2VA;
+			      if (ep1 && ep3) type = HP_QUAD_0E_2VB;
+			      break;
+			    }
+			  case 3: 
+			    if (!ep4) type = HP_QUAD_0E_3V; 
+			    break; 
+			  case 4: 
+			    type = HP_QUAD_0E_4V; 
+			    break; 
+			  }
+			break;
+		      }
+		    case 1:
+		      {
+			if (isedge1)
+			  {
+			    switch (cp1+cp2+ep3+ep4)
+			      {
+			      case 0: 
+				type = HP_QUAD_SINGEDGE; 
+				break;
+			      case 1:
+				{
+				  if (cp1) type = HP_QUAD_1E_1VA;
+				  if (cp2) type = HP_QUAD_1E_1VB;
+				  if (ep3) type = HP_QUAD_1E_1VC;
+				  if (ep4) type = HP_QUAD_1E_1VD; 
+				  break; 
+				}
+			      case 2:
+				{
+				  if (cp1 && cp2) type = HP_QUAD_1E_2VA; 
+				  if (cp1 && ep3) type = HP_QUAD_1E_2VB; 
+				  if (cp1 && ep4) type = HP_QUAD_1E_2VC; 
+				  if (cp2 && ep3) type = HP_QUAD_1E_2VD; 
+				  if (cp2 && ep4) type = HP_QUAD_1E_2VE; 
+				  if (ep3 && ep4) type = HP_QUAD_1E_2VF; 
+				  break; 
+				}
+			      case 3:
+				{
+				  if (cp1 && cp2 && ep3) type = HP_QUAD_1E_3VA;
+				  if (cp1 && cp2 && ep4) type = HP_QUAD_1E_3VB;
+				  if (cp1 && ep3 && ep4) type = HP_QUAD_1E_3VC;
+				  if (cp2 && ep3 && ep4) type = HP_QUAD_1E_3VD;
+				  break;
+				}
+			      case 4:
+				{
+				  type = HP_QUAD_1E_4V; 
+				  break;
+				}
+			      }
+			  }
+			break;
+		      }
+		    case 2:
+		      {
+			if (isedge1 && isedge4)
+			  {
+			    if (!cp2 && !ep3 && !cp4)
+			      type = HP_QUAD_2E;
+			  
+			    if (cp2 && !ep3 && !cp4)
+			      type = HP_QUAD_2E_1VA;
+			    if (!cp2 && ep3 && !cp4)
+			      type = HP_QUAD_2E_1VB;
+			    if (!cp2 && !ep3 && cp4)
+			      type = HP_QUAD_2E_1VC;
+
+			    if (cp2 && ep3 && !cp4)
+			      type = HP_QUAD_2E_2VA;
+			    if (cp2 && !ep3 && cp4)
+			      type = HP_QUAD_2E_2VB;
+			    if (!cp2 && ep3 && cp4)
+			      type = HP_QUAD_2E_2VC;
+
+			    if (cp2 && ep3 && cp4)
+			      type = HP_QUAD_2E_3V;
+			  }
+			if (isedge1 && isedge3)
+			  {
+			    switch (sumcp)
+			      {
+			      case 0: 
+				type = HP_QUAD_2EB_0V; break;
+			      case 1:
+				{
+				  if (cp1) type = HP_QUAD_2EB_1VA; 
+				  if (cp2) type = HP_QUAD_2EB_1VB; 
+				  break;
+				}
+			      case 2:
+				{
+				  if (cp1 && cp2) { type = HP_QUAD_2EB_2VA; }
+				  if (cp1 && cp3) { type = HP_QUAD_2EB_2VB; }
+				  if (cp1 && cp4) { type = HP_QUAD_2EB_2VC; }
+				  if (cp2 && cp4) { type = HP_QUAD_2EB_2VD; }
+				  break;
+				}
+			      case 3:
+				{
+				  if (cp1 && cp2 && cp3) { type = HP_QUAD_2EB_3VA; }
+				  if (cp1 && cp2 && cp4) { type = HP_QUAD_2EB_3VB; }
+				  break;
+				}
+			      case 4:
+				{
+				  type = HP_QUAD_2EB_4V; break;
+				}
+			      }
+			  }
+			break;
+		      }
 
+		    case 3:
+		      {
+			if (isedge1 && isedge2 && isedge4)
+			  {
+			    if (!cp3 && !cp4) type = HP_QUAD_3E;
+			    if (cp3 && !cp4) type = HP_QUAD_3E_3VA;
+			    if (!cp3 && cp4) type = HP_QUAD_3E_3VB;
+			    if (cp3 && cp4) type = HP_QUAD_3E_4V;
+			  }
+			break;
+		      }
 
+		    case 4:
+		      {
+			type = HP_QUAD_4E;
+			break;
+		      }
+		    }
 
+		  if (type != HP_NONE)
+		    {
+		      pnums[0] = el.PNumMod (j);
+		      pnums[1] = el.PNumMod (j+1);
+		      pnums[2] = el.PNumMod (j+2);
+		      pnums[3] = el.PNumMod (j+3);
+		      break;
+		    }
+		}
+	      if (type == HP_NONE)
+		{
+		  (*testout) << "undefined element" << endl
+			     << "cp = " << cp1 << cp2 << cp3 << cp4 << endl
+			     << "ep = " << ep1 << ep2 << ep3 << ep4 << endl
+			     << "isedge = " << isedge1 << isedge2 << isedge3 
+			     << isedge4 << endl;
+		}
+	      break;
+	    }
+	  }
 
+	if (type == HP_NONE)
+	  {
+	    cerr << "undefined QUAD type" << endl;
+	    for (j = 0; j < 4; j++)
+	      pnums[j] = el[j];
+	  }
 
+	HPRefElement hpel;
+	hpel.type = type;
+	for (j = 0; j < 8; j++)
+	  hpel.pnums[j] = pnums[j];
+	hpel.index = el.GetIndex();
+	hpel.level = 1;
+	hpel.coarse_elnr = i-1;
 
 
+	static const double points[3][2] = 
+	  { { 1, 0 }, { 0, 1 }, { 0, 0 } };
+	for (j = 0; j < 3; j++)
+	  for (k = 0; k < 3; k++)
+	    if (pnums[j] == el[k])
+	      {
+		hpel.param[j][0] = points[k][0];
+		hpel.param[j][1] = points[k][1];
+	      }
+	elements.Append (hpel);
+      }
+  
+  
+  
 
+  
 
+    for (int i = 1; i <= mesh.GetNSeg(); i++)
+      {
+	Segment & seg = mesh.LineSegment(i);
+      
+	HPREF_ELEMENT_TYPE type = HP_NONE;
 
+	int cp1 = cornerpoint.Test (seg.p1);
+	int cp2 = cornerpoint.Test (seg.p2);
 
+	INDEX_2 i2;
+	i2 = INDEX_2(seg.p1, seg.p2);
+	i2.Sort();
+	if (!edges.Used (i2))
+	  {
+	    cp1 = edgepoint.Test (seg.p1);
+	    cp2 = edgepoint.Test (seg.p2);
+	  }
 
+	HPRefElement hpel;
+	hpel.index = seg.edgenr + 10000 * seg.si;
+	hpel.level = 1;
+      
+	hpel.pnums[0] = seg.p1;
+	hpel.pnums[1] = seg.p2;
+	hpel.param[0][0] = seg.epgeominfo[0].dist;
+	hpel.param[1][0] = seg.epgeominfo[1].dist;
+      
+	if (!cp1 && !cp2)
+	  hpel.type = HP_SEGM;
+	else if (cp1 && !cp2)
+	  hpel.type = HP_SEGM_SINGCORNERL;
+	else if (!cp1 && cp2)
+	  hpel.type = HP_SEGM_SINGCORNERR;
+	else
+	  hpel.type = HP_SEGM_SINGCORNERS;
 
+	(*testout) << "refine segment " << seg << ", hptype= " << hpel.type << endl;
 
+	elements.Append (hpel);
 
+	(*testout) << "add seg: " << endl << seg << endl;
+	(*testout) << "param = " << elements.Last().param[0][0] << ", " << elements.Last().param[1][0] << endl;
+      }
+  }
 
-/* *********** Tet - Refinement - 1 edge *************** */
+  
 
+  /* *******************************  DoRefinement *************************************** */
 
+  // parameter "fine": false ... divide elements by a factor of 3/4 to 1/4
+  //                   true  ... divide by 7/8 to 1/8
 
-// HP_TET_1E_0V
-int reftet_1e_0v_splitedges[][3] =
-{
-  { 1, 3, 5 },
-  { 1, 4, 6 },
-  { 2, 3, 7 },
-  { 2, 4, 8 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftet_1e_0v_newelstypes[] =
-{
-  HP_PRISM_SINGEDGE,
-  HP_PRISM,
-  HP_NONE,
-};
-int reftet_1e_0v_newels[][8] =
-{
-  { 1, 5, 6, 2, 7, 8 },
-  { 7, 3, 5, 8, 4, 6 }
-};
-HPRef_Struct reftet_1e_0v =
-{
-  HP_TET,
-  reftet_1e_0v_splitedges, 
-  0, 0,
-  reftet_1e_0v_newelstypes, 
-  reftet_1e_0v_newels
-};
+  void DoRefinement (Mesh & mesh, ARRAY<HPRefElement> & elements,
+		     Refinement * ref, bool fine) 
+  {
+    INDEX_2_HASHTABLE<int> newpts(elements.Size()+1);
+    INDEX_3_HASHTABLE<int> newfacepts(elements.Size()+1);
+  
+    // prepare new points
+  
+    int oldelsize = elements.Size();
+    for (int i = 0; i < oldelsize; i++)
+      {
+	HPRefElement & el = elements[i];
+	HPRef_Struct * hprs = Get_HPRef_Struct (el.type);
 
+	if (!hprs)
+	  {
+	    cout << "Refinementstruct not defined for element " << el.type << endl;
+	    continue;
+	  }
 
+	int j = 0;
+	while (hprs->splitedges[j][0])
+	  {
+	    INDEX_2 i2(el.pnums[hprs->splitedges[j][0]-1],
+		       el.pnums[hprs->splitedges[j][1]-1]);
+	    if (!newpts.Used (i2))
+	      {
+		Point3d np = Center (mesh.Point (i2.I1()),
+				     mesh.Point (i2.I2()));
+		np = Center (mesh.Point (i2.I1()),np);
+		if ( fine ) np = Center (mesh.Point (i2.I1()),np);
 
+		int npi = mesh.AddPoint (np);
+		newpts.Set (i2, npi);
+	      }
+	    j++;
+	  }
 
 
-// HP_TET_1E_1VA
-int reftet_1e_1va_splitedges[][3] =
-{
-  { 1, 3, 5 },
-  { 1, 4, 6 },
-  { 2, 3, 7 },
-  { 2, 4, 8 },
-  { 1, 2, 9 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftet_1e_1va_newelstypes[] =
-{
-  HP_TET_1E_1VA,
-  HP_PRISM_SINGEDGE,
-  HP_PRISM,
-  HP_NONE,
-};
-int reftet_1e_1va_newels[][8] =
-{
-  { 1, 9, 5, 6 },
-  { 9, 5, 6, 2, 7, 8 },
-  { 7, 3, 5, 8, 4, 6 }
-};
-HPRef_Struct reftet_1e_1va =
-{
-  HP_TET,
-  reftet_1e_1va_splitedges, 
-  0, 0,
-  reftet_1e_1va_newelstypes, 
-  reftet_1e_1va_newels
-};
+	j = 0;
+	if (hprs->splitfaces)
+	  while (hprs->splitfaces[j][0])
+	    {
+	      INDEX_3 i3(el.pnums[hprs->splitfaces[j][0]-1],
+			 el.pnums[hprs->splitfaces[j][1]-1],
+			 el.pnums[hprs->splitfaces[j][2]-1]);
 
+	      if (i3.I2() > i3.I3()) Swap (i3.I2(), i3.I3());
 
+	      if (!newfacepts.Used (i3))
+		{
+		  Point3d np = Center (mesh.Point (i3.I2()),
+				       mesh.Point (i3.I3()));
+		  np = Center (mesh.Point (i3.I1()),np);
+		  if ( fine ) np = Center (mesh.Point (i3.I1()),np);
+		  int npi = mesh.AddPoint (np);
+		  newfacepts.Set (i3, npi);
+		}
+	      j++;
+	    }
+      }
+  
+    for (int i = 0; i < oldelsize; i++)
+      {
+	HPRefElement & el = elements[i];
+	HPRef_Struct * hprs = Get_HPRef_Struct (el.type);
+	int newlevel = el.level + 1;
+
+	if (el.type == HP_SEGM ||
+	    el.type == HP_TRIG ||
+	    el.type == HP_QUAD ||
+	    el.type == HP_TET ||
+	    el.type == HP_PRISM ||
+	    el.type == HP_HEX)
+	  newlevel = el.level;
+
+	if (!hprs) continue;
+
+	int newpnums[64];
+	double newparam[64][3];
+
+	int j;
+	for (j = 0; j < 8; j++)
+	  {
+	    newpnums[j] = el.pnums[j];
+	    for (int l = 0; l < 3; l++)
+	      newparam[j][l] = el.param[j][l];
+	  }
 
 
+	// split edges, incl. transferring curvature
+	j = 0;
+	while (hprs->splitedges[j][0])
+	  {
+	    INDEX_2 i2(el.pnums[hprs->splitedges[j][0]-1],
+		       el.pnums[hprs->splitedges[j][1]-1]);
 
+	    int npi = newpts.Get(i2);
+	    newpnums[hprs->splitedges[j][2]-1] = npi;
 
-// HP_TET_1E_1VB
-int reftet_1e_1vb_splitedges[][3] =
-{
-  { 1, 3, 5 },
-  { 1, 4, 6 },
-  { 2, 3, 7 },
-  { 2, 4, 8 },
-  { 4, 1, 9 },
-  { 4, 2, 10 },
-  { 4, 3, 11 },
-  { 0, 0, 0 }
-};
-int reftet_1e_1vb_splitelements[][5] =
-{
-  { 4, 1, 2, 3, 12 },
-  { 0 }
-};
+	    double fac1, fac2;
+	    if ( fine ) { fac1 = 0.875; fac2 = 0.125; }
+	    else { fac1 = 0.75; fac2 = 0.25; }
 
-HPREF_ELEMENT_TYPE reftet_1e_1vb_newelstypes[] =
-{
-  HP_PRISM_SINGEDGE,
-  HP_TET_0E_1V,
-  HP_PYRAMID,
-  HP_TET,
-  HP_PYRAMID, 
-  HP_TET,
-  HP_PYRAMID,
-  HP_TET,
-  HP_PYRAMID,
-  HP_TET,
-  HP_NONE,
-};
-int reftet_1e_1vb_newels[][8] =
-{
-  { 1, 5, 6, 2, 7, 8 },
-  { 4, 11, 10, 9 },
-  { 7, 8, 10, 11, 12 },
-  { 3, 7, 11, 12 },
-  { 5, 11, 9, 6, 12 },
-  { 5, 3, 11, 12 },
-  { 6, 9, 10, 8, 12 },
-  { 5, 7, 3, 12 },
-  { 5, 6, 8, 7, 12 },
-  { 9, 11, 10, 12 }
-};
-HPRef_Struct reftet_1e_1vb =
-{
-  HP_TET,
-  reftet_1e_1vb_splitedges, 
-  0,
-  reftet_1e_1vb_splitelements, 
-  reftet_1e_1vb_newelstypes, 
-  reftet_1e_1vb_newels
-};
+	    for (int l = 0; l < 3; l++)
+	      newparam[hprs->splitedges[j][2]-1][l] =
+		fac1 * el.param[hprs->splitedges[j][0]-1][l] + 
+		fac2 * el.param[hprs->splitedges[j][1]-1][l];
+	      
+	    j++;
+	  }
 
+	// split faces
+	j = 0;
+	if (hprs->splitfaces)
+	  while (hprs->splitfaces[j][0])
+	    {
+	      INDEX_3 i3(el.pnums[hprs->splitfaces[j][0]-1],
+			 el.pnums[hprs->splitfaces[j][1]-1],
+			 el.pnums[hprs->splitfaces[j][2]-1]);
+	      if (i3.I2() > i3.I3())
+		Swap (i3.I2(), i3.I3());
+	      int npi = newfacepts.Get(i3);
+	      newpnums[hprs->splitfaces[j][3]-1] = npi;
+
+	      double fac1, fac2;
+	      if ( fine ) { fac1 = 0.75; fac2 = 0.125; }
+	      else { fac1 = 0.5; fac2 = 0.25; }
+
+	      for (int l = 0; l < 3; l++)
+		newparam[hprs->splitfaces[j][3]-1][l] =
+		  fac1 * el.param[hprs->splitfaces[j][0]-1][l] + 
+		  fac2 * el.param[hprs->splitfaces[j][1]-1][l] + 
+		  fac2 * el.param[hprs->splitfaces[j][2]-1][l];
+	      j++;
+	    }
 
+	// split elements
+	j = 0;
+	if (hprs->splitelements)
+	  while (hprs->splitelements[j][0])
+	    {
+	      int pi1 = el.pnums[hprs->splitelements[j][0]-1];
+	      Point3d np = 
+		Center (Center (mesh.Point (pi1),
+				mesh.Point (el.pnums[hprs->splitelements[j][1]-1])),
+			Center (mesh.Point (el.pnums[hprs->splitelements[j][2]-1]),
+				mesh.Point (el.pnums[hprs->splitelements[j][3]-1])));
+	      // divide once more, if fine subdivision is wanted
+	      if ( fine ) np = Center (mesh.Point (pi1),np);
+	      int npi = mesh.AddPoint (np);
+	      newpnums[hprs->splitelements[j][4]-1] = npi;
 
+	      double fac1, fac2;
+	      if ( fine ) { fac1 = 0.625; fac2 = 0.125; }
+	      else { fac1 = 0.25; fac2 = 0.25; }
 
+	      for (int l = 0; l < 3; l++)
+		newparam[hprs->splitelements[j][4]-1][l] =
+		  fac1 * el.param[hprs->splitelements[j][0]-1][l] + 
+		  fac2 * el.param[hprs->splitelements[j][1]-1][l] + 
+		  fac2 * el.param[hprs->splitelements[j][2]-1][l] + 
+		  fac2 * el.param[hprs->splitelements[j][3]-1][l];
 
+	      j++;
+	    }
 
+	j = 0;
+	while (hprs->neweltypes[j])
+	  {
+	    HPRefElement newel;
+	    newel.type = hprs->neweltypes[j];
+	    for (int k = 0; k < 8; k++)
+	      newel.pnums[k] = newpnums[hprs->newels[j][k]-1];
 
+	    newel.index = elements[i].index;
+	    newel.coarse_elnr = elements[i].coarse_elnr;
+	    newel.level = newlevel;
 
-// HP_TET_1E_2VA
-int reftet_1e_2va_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 1, 8 },
-  { 2, 3, 9 },
-  { 2, 4, 10 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftet_1e_2va_newelstypes[] =
-{
-  HP_TET_1E_1VA,
-  HP_TET_1E_1VA,
-  HP_PRISM_SINGEDGE,
-  HP_PRISM,
-  HP_NONE,
-};
-int reftet_1e_2va_newels[][8] =
-{
-  { 1, 5, 6, 7 },
-  { 2, 8, 10, 9 },
-  { 5, 6, 7, 8, 9, 10 },
-  { 4, 10, 7, 3, 9, 6 },
-};
-HPRef_Struct reftet_1e_2va =
-{
-  HP_TET,
-  reftet_1e_2va_splitedges, 
-  0, 0,
-  reftet_1e_2va_newelstypes, 
-  reftet_1e_2va_newels
-};
+	    for (int k = 0; k < 8; k++)  
+	      for (int l = 0; l < 3; l++)
+		newel.param[k][l] = newparam[hprs->newels[j][k]-1][l];
 
+	    if (j == 0) 
+	      elements[i] = newel;
+	    else        
+	      elements.Append (newel);
 
+	    j++;
+	  }
+      }
+  }
 
 
 
 
 
-// HP_TET_1E_2VB
-int reftet_1e_2vb_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 3, 8 },
-  { 2, 4, 9 },
-  { 3, 1, 10 },
-  { 3, 2, 11 },
-  { 3, 4, 12 },
-  { 0, 0, 0 }
-};
-int reftet_1e_2vb_splitelements[][5] =
-{
-  { 3, 4, 1, 2, 13 },
-  { 0 }
-};
 
-HPREF_ELEMENT_TYPE reftet_1e_2vb_newelstypes[] =
-{
-  HP_TET_1E_1VA,
-  HP_PRISM_SINGEDGE,
-  HP_TET_0E_1V,
-  HP_PYRAMID,
-  HP_TET,
-  HP_PYRAMID, 
-  HP_TET,
-  HP_PYRAMID,
-  HP_TET,
-  HP_PYRAMID,
-  HP_TET,
-  HP_NONE,
-};
-int reftet_1e_2vb_newels[][8] =
-{
-  { 1, 5, 6, 7 },
-  { 5, 6, 7, 2, 8, 9 },
-  { 3, 10, 11, 12 },
-
-  { 8, 9, 12, 11, 13 },
-  { 4, 12, 9, 13 },
-  { 6, 10, 12, 7, 13 },
-  { 4, 7, 12, 13 },
-  { 6, 8, 11, 10, 13 },
-  { 4, 9, 7, 13 },
-  { 6, 7, 9, 8, 13 },
-  { 10, 11, 12, 13 },
-};
-HPRef_Struct reftet_1e_2vb =
-{
-  HP_TET,
-  reftet_1e_2vb_splitedges, 
-  0,
-  reftet_1e_2vb_splitelements, 
-  reftet_1e_2vb_newelstypes, 
-  reftet_1e_2vb_newels
-};
+  /* ************************** DoRefineDummies ******************************** */
 
+  void DoRefineDummies (Mesh & mesh, ARRAY<HPRefElement> & elements,
+			Refinement * ref)
+  {
+    int oldelsize = elements.Size();
+
+    for (int i = 0; i < oldelsize; i++)
+      {
+	HPRefElement & el = elements[i];
+	HPRef_Struct * hprs = Get_HPRef_Struct (el.type);
+	if (!hprs) continue;
+
+	if (el.type != HP_DUMMY_QUAD_SINGCORNER &&
+	    el.type != HP_PYRAMID_EDGES &&
+	    el.type != HP_PYRAMID_0E_1V &&
+	    el.type != HP_HEX_0E_1V &&
+	    el.type != HP_HEX_1E_1V &&
+	    el.type != HP_HEX_1E_0V &&
+	    el.type != HP_HEX_3E_0V
+	    ) continue;
+
+	int newlevel = el.level;
+
+	int newpnums[8];
+	int j;
+	for (j = 0; j < 8; j++)
+	  newpnums[j] = el.pnums[j];
+
+	double newparam[8][3];
+	for (j = 0; j < 8; j++)
+	  for (int k = 0; k < 3; k++)
+	    newparam[j][k] = el.param[j][k];
+
+	j = 0;
+	while (hprs->neweltypes[j])
+	  {
+	    HPRefElement newel;
+	    newel.type = hprs->neweltypes[j];
+	    for (int k = 0; k < 8; k++)
+	      newel.pnums[k] = newpnums[hprs->newels[j][k]-1];
+	    newel.index = el.index;
+	    newel.coarse_elnr = el.coarse_elnr;
+	    newel.level = newlevel;
+
+	    for (int k = 0; k < 8; k++)
+	      for (int l = 0; l < 3; l++)
+		newel.param[k][l] = newparam[hprs->newels[j][k]-1][l];
+		
+	    if (j == 0)
+	      elements[i] = newel;
+	    else
+	      elements.Append (newel);
+	    j++;
+	  }
+      }
+  }
 
 
 
 
 
-// HP_TET_1E_2VC
-int reftet_1e_2vc_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 3, 8 },
-  { 2, 4, 9 },
-  { 4, 1, 10 },
-  { 4, 2, 11 },
-  { 4, 3, 12 },
-  { 0, 0, 0 }
-};
-int reftet_1e_2vc_splitelements[][5] =
-{
-  { 4, 1, 2, 3, 13 },
-  { 0 }
-};
 
-HPREF_ELEMENT_TYPE reftet_1e_2vc_newelstypes[] =
-{
-  HP_TET_1E_1VA,
-  HP_PRISM_SINGEDGE,
-  HP_TET_0E_1V,
-  HP_PYRAMID,
-  HP_TET,
-  HP_PYRAMID, 
-  HP_TET,
-  HP_PYRAMID,
-  HP_TET,
-  HP_PYRAMID,
-  HP_TET,
-  HP_NONE,
-};
-int reftet_1e_2vc_newels[][8] =
-{
-  { 1, 5, 6, 7 },
-  { 5, 6, 7, 2, 8, 9 },
-  { 4, 11, 10, 12 },
-  { 8, 9, 11, 12, 13 },
-  { 3, 8, 12, 13 },
-  { 7, 6, 12, 10, 13 },
-  { 3, 12, 6, 13 },
-  { 9, 7, 10, 11, 13 },
-  { 3, 6, 8, 13 },
-  { 6, 7, 9, 8, 13 },
-  { 10, 12, 11, 13 }
-};
-HPRef_Struct reftet_1e_2vc =
-{
-  HP_TET,
-  reftet_1e_2vc_splitedges, 
-  0,
-  reftet_1e_2vc_splitelements, 
-  reftet_1e_2vc_newelstypes, 
-  reftet_1e_2vc_newels
-};
 
+  void SubdivideDegeneratedHexes (Mesh & mesh, ARRAY<HPRefElement> & elements)
+  {
+    int oldne = elements.Size();
+    for (int i = 0; i < oldne; i++)
+      if (Get_HPRef_Struct (elements[i].type)->geom == HP_HEX)
+	{
+	  bool common = 0;
+	  for (int j = 0; j < 8; j++)
+	    for (int k = 0; k < j; k++)
+	      if (elements[i].pnums[j] == elements[i].pnums[k])
+		common = 1;
+	  if (common)
+	    {
+	      HPRefElement el = elements[i];
 
+	      Point<3> center(0,0,0);
+	      double newparam[3] = { 0, 0, 0 };
 
+	      for (int j = 0; j < 8; j++)
+		{
+		  center += 0.125 * Vec<3> (mesh[el.pnums[j]]);
+		  for (int l = 0; l < 3; l++)
+		    newparam[l] += 0.125 * el.param[j][l];
+		}
 
+	      int npi = mesh.AddPoint (center);
 
+	      const ELEMENT_FACE * faces = MeshTopology::GetFaces (HEX);
 
+	      for (int j = 0; j < 6; j++)
+		{
+		  ARRAY<int> pts;
+		  for (int k = 0; k < 4; k++)
+		    {
+		      bool same = 0;
+		      for (int l = 0; l < pts.Size(); l++)
+			if (el.pnums[pts[l]] == el.pnums[faces[j][k]-1])
+			  same = 1;
+		      if (!same)
+			pts.Append (faces[j][k]-1);
 
+		    }
+		  
+		  HPRefElement newel = el;
+		  if (pts.Size() == 3)
+		    {
+		      for (int k = 0; k < 3; k++)
+			{
+			  newel.pnums[k] = el.pnums[pts[2-k]];
+			  for (int l = 0; l < 3; l++)
+			    newel.param[k][l] = el.param[pts[2-k]][l];
+			}
+		      newel.pnums[3] = npi;
+		      for (int l = 0; l < 3; l++)
+			newel.param[3][l] = newparam[l];
+
+		      newel.type = HP_TET;
+		    }
+		  else
+		    {
+		      for (int k = 0; k < 4; k++)
+			{
+			  newel.pnums[k] = el.pnums[pts[3-k]];
+			  for (int l = 0; l < 3; l++)
+			    newel.param[k][l] = el.param[pts[3-k]][l];
+			}
+
+		      newel.pnums[4] = npi;
+		      for (int l = 0; l < 3; l++)
+			newel.param[4][l] = newparam[l];
+
+		      newel.type = HP_PYRAMID;
+		    }
+		  
+		  if (j == 0)
+		    elements[i] = newel;
+		  else
+		    elements.Append (newel);
+		}
+	    }
+	}
+  }
 
 
+  void CalcStatistics (ARRAY<HPRefElement> & elements)
+  {
+    return;
+    
+    int i, p;
+    int nsegm = 0, ntrig = 0, nquad = 0;
+    int nhex = 0, nprism = 0, npyramid = 0, ntet = 0;
+    int maxlevel = 0;
+
+    for (i = 1; i <= elements.Size(); i++)
+      {
+	const HPRefElement & el = elements.Get(i);
+	maxlevel = max2 (el.level, maxlevel);
+	switch (Get_HPRef_Struct (el.type)->geom)
+	  {
+	  case HP_SEGM:
 
-// HP_TET_1E_2VD
-int reftet_1e_2vd_splitedges[][3] =
-{
-  { 1, 3, 5 },
-  { 1, 4, 6 },
-  { 2, 3, 7 },
-  { 2, 4, 8 },
-  { 3, 1, 9 },
-  { 3, 2, 10 },
-  { 3, 4, 11 },
-  { 4, 1, 12 },
-  { 4, 2, 13 },
-  { 4, 3, 14 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE reftet_1e_2vd_newelstypes[] =
-{
-  HP_PRISM_SINGEDGE,
-  HP_TET_0E_1V,
-  HP_TET_0E_1V,
-  HP_PRISM,
-  HP_HEX,
-  HP_NONE,
-};
-int reftet_1e_2vd_newels[][8] =
-{
-  { 1, 5, 6, 2, 7, 8 },
-  { 4, 13, 12, 14 },
-  { 3, 10, 11, 9 },
-  { 14, 13, 12, 11, 10, 9 },
-  { 6, 12, 13, 8, 5, 9, 10, 7 },
-};
-HPRef_Struct reftet_1e_2vd =
-{
-  HP_TET,
-  reftet_1e_2vd_splitedges, 
-  0, 0,
-  reftet_1e_2vd_newelstypes, 
-  reftet_1e_2vd_newels
-};
+	    {
+	      nsegm++;
+	      break;
+	    }
+	  case HP_TRIG:
+	    {
+	      ntrig ++;
+	      break;
+	    }
+	  case HP_QUAD:
+	    {
+	      nquad++;
+	      break;
+	    }
+	  case HP_TET:
+	    {
+	      ntet++;
+	      break;
+	    }
 
+	  case HP_PRISM:
+	    {
+	      nprism++;
+	      break;
+	    }
 
+	  case HP_PYRAMID:
+	    {
+	      npyramid++;
+	      break;
+	    }
 
+	  case HP_HEX:
+	    {	
+	      nhex++;
+	      break;
+	    }
 
+	  default:
+	    {
+	      cerr << "statistics error, unknown element type" << endl;
+	    }
+	  }
+      }
+
+    cout << "level = " << maxlevel << endl;
+    cout << "nsegm = " << nsegm << endl;
+    cout << "ntrig = " << ntrig << ", nquad = " << nquad << endl;
+    cout << "ntet = " << ntet << ", npyr = " << npyramid
+	 << ", nprism = " << nprism << ", nhex = " << nhex << endl;
+
+    return;
+
+    double memcost = 0, cpucost = 0;
+    for (p = 1; p <= 20; p++)
+      {
+	memcost = (ntet + nprism + nhex) * pow (static_cast<double>(p), 6.0);
+	cpucost = (ntet + nprism + nhex) * pow (static_cast<double>(p), 9.0);
+	cout << "costs for p = " << p << ": mem = " << memcost << ", cpu = " << cpucost << endl;
+      }
+
+    double memcosttet = 0;
+    double memcostprism = 0;
+    double memcosthex = 0;
+    double memcostsctet = 0;
+    double memcostscprism = 0;
+    double memcostschex = 0;
+    double cpucosttet = 0;
+    double cpucostprism = 0;
+    double cpucosthex = 0;
+
+    for (i = 1; i <= elements.Size(); i++)
+      {
+	const HPRefElement & el = elements.Get(i);
+	switch (el.type)
+	  {
+	  case HP_TET:
+	  case HP_TET_0E_1V:
+	  case HP_TET_1E_0V:
+	  case HP_TET_1E_1VA:
+	    {
+	      int p1 = maxlevel - el.level + 1;
+	      (*testout) << "p1 = " << p1 << ", P1^6 = " << pow (static_cast<double>(p1), 6.0)
+			 << " (p1-3)^6 = " << pow ( static_cast<double>(max2(p1-3, 0)), 6.0) 
+			 << " p1^3 = " << pow ( static_cast<double>(p1), 3.0) 
+			 << " (p1-3)^3 = " << pow ( static_cast<double>(p1-3), 3.0) 
+			 << " [p1^3-(p1-3)^3]^2 = " << sqr (pow (static_cast<double>(p1),3.0) - pow ( static_cast<double>(p1-3), 3.0))
+			 << endl;
+
+	      p1 /= 2 +1;
+	      memcosttet += pow (static_cast<double>(p1), 6.0);
+	      memcostsctet += pow (static_cast<double>(p1), 6.0) - pow ( static_cast<double>(max2(p1-3, 1)), 6.0);
+	      cpucosttet += pow (static_cast<double>(p1), 9.0);
+	      break;
+	    }
+	  case HP_PRISM:
+	  case HP_PRISM_SINGEDGE:
+	    {
+	      int p1 = maxlevel - el.level + 1;
+	      p1 /= 2 +1;
+	      memcostprism += pow (static_cast<double>(p1), 6.0);
+	      memcostscprism += pow (static_cast<double>(p1), 6.0) - pow ( static_cast<double>(max2(p1-3, 1)), 6.0);
+	      cpucostprism += pow (static_cast<double>(p1), 9.0);
+	      break;
+	    }
+	  case HP_HEX:
+	    {	
+	      int p1 = maxlevel - el.level + 1;
+	      int p2 = maxlevel;
+	      p1 /= 2 +1;
+	      p2 /= 2 +1;
+	      memcosthex += pow (static_cast<double>(p1), 4.0) * pow (static_cast<double>(p2), 2.0);
+	      memcostschex += pow (static_cast<double>(p1), 6.0) - pow ( static_cast<double>(max2(p1-2, 0)), 6.0);
+	      cpucosthex += pow (static_cast<double>(p1), 6.0) * pow (static_cast<double>(p2), 3.0);
+	      break;
+	    }
+	  default:
+	    ;
+	  }
+      }
+    cout << "TET: hp-memcost = " << memcosttet 
+	 << ", scmemcost = " << memcostsctet
+	 << ", cpucost = " << cpucosttet
+	 << endl;
+    cout << "PRI: hp-memcost = " << memcostprism
+	 << ", scmemcost = " << memcostscprism
+	 << ", cpucost = " << cpucostprism << endl;
+    cout << "HEX: hp-memcost = " << memcosthex
+	 << ", scmemcost = " << memcostschex
+	 << ", cpucost = " << cpucosthex << endl;
+  }
 
 
 
@@ -1626,2078 +2451,155 @@ HPRef_Struct reftet_1e_2vd =
 
 
 
+  /* ***************************** HPRefinement ********************************** */
 
-//  HP_TET_2EA_0V,  // 2 edges connected
-int reftet_2ea_0v_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 3, 9 },
-  { 2, 4, 10 },
-  { 3, 2, 12 },
-  { 3, 4, 13 },
-  { 0, 0, 0 }
-};
-int reftet_2ea_0v_splitfaces[][4] =
-  {
-    { 1, 2, 3, 17 },
-    { 0, 0, 0, 0 }
-  };
-HPREF_ELEMENT_TYPE reftet_2ea_0v_newelstypes[] =
+  void HPRefinement (Mesh & mesh, Refinement * ref, int levels)
   {
-    HP_PYRAMID_EDGES,
-    HP_PRISM_SINGEDGE,
-    HP_PRISM_SINGEDGE,
-    HP_PRISM,
-    HP_TET,
-    HP_NONE,
-  };
-int reftet_2ea_0v_newels[][8] =
-{
-  { 1, 5, 17, 6, 7 },
-  { 5, 17, 7, 2, 9, 10 },
-  { 6, 7, 17, 3, 13, 12 },
-  { 17, 9, 12, 7, 10, 13 },
-  { 7, 10, 13, 4 },
-};
-HPRef_Struct reftet_2ea_0v =
-{
-  HP_TET,
-  reftet_2ea_0v_splitedges, 
-  reftet_2ea_0v_splitfaces, 
-  0,
-  reftet_2ea_0v_newelstypes, 
-  reftet_2ea_0v_newels
-};
+    PrintMessage (1, "HP Refinement called, levels = ", levels);
 
+    // ARRAY<HPRefElement> hpelements;
 
+    delete mesh.hpelements;
+    mesh.hpelements = new ARRAY<HPRefElement>;
 
+    mesh.coarsemesh = new Mesh;
+    *mesh.coarsemesh = mesh;
 
-
-
-//  HP_TET_2EA_1VB, 
-int reftet_2ea_1vb_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 3, 9 },
-  { 2, 4, 10 },
-  { 3, 1, 11 },
-  { 3, 2, 12 },
-  { 3, 4, 13 },
-  { 0, 0, 0 }
-};
-int reftet_2ea_1vb_splitfaces[][4] =
-  {
-    { 1, 2, 3, 17 },
-    { 0, 0, 0, 0 }
-  };
-HPREF_ELEMENT_TYPE reftet_2ea_1vb_newelstypes[] =
-  {
-    HP_PYRAMID_EDGES,
-    HP_TET_1E_1VA,
-    HP_PRISM_SINGEDGE,
-    HP_PRISM_SINGEDGE,
-    HP_PRISM,
-    HP_TET,
-    HP_NONE,
-  };
-int reftet_2ea_1vb_newels[][8] =
-{
-  { 1, 5, 17, 6, 7 },
-  { 3, 11, 12, 13 },
-  { 5, 17, 7, 2, 9, 10 },
-  { 6, 7, 17, 11, 13, 12 },
-  { 17, 9, 12, 7, 10, 13 },
-  { 7, 10, 13, 4 },
-};
-HPRef_Struct reftet_2ea_1vb =
-{
-  HP_TET,
-  reftet_2ea_1vb_splitedges, 
-  reftet_2ea_1vb_splitfaces, 
-  0,
-  reftet_2ea_1vb_newelstypes, 
-  reftet_2ea_1vb_newels
-};
-
+    ARRAY<HPRefElement> & hpelements = *mesh.hpelements;
 
 
 
+    PrepareElements (mesh, hpelements);
 
+    for (int j = 1; j <= levels; j++)
+      {
+	cout << "hp-refine level " << j << flush;
+	// last parameter: true for fine subdivision (7/8 : 1/8), false for 3/4 : 1/4
+	// Please check into CVS only with this parameter set to "true"!
+	DoRefinement (mesh, hpelements, ref, true); // FB 
+	DoRefineDummies (mesh, hpelements, ref);
+	CalcStatistics (hpelements);
+	cout << " done" << endl;
+      }
 
+    SubdivideDegeneratedHexes (mesh, hpelements);
 
+    mesh.ClearSegments();
+    mesh.ClearSurfaceElements();
+    mesh.ClearVolumeElements();
 
-
-//  HP_TET_2EA_3V,  // 2 edges connected
-int reftet_2ea_3v_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 1, 8 },
-  { 2, 3, 9 },
-  { 2, 4, 10 },
-  { 3, 1, 11 },
-  { 3, 2, 12 },
-  { 3, 4, 13 },
-  { 4, 1, 14 },
-  { 4, 2, 15 },
-  { 4, 3, 16 },
-  { 0, 0, 0 }
-};
-int reftet_2ea_3v_splitfaces[][4] =
-  {
-    { 1, 2, 3, 17 },
-    { 2, 3, 4, 18 },
-    { 3, 4, 2, 19 },
-    { 4, 2, 3, 20 },
-    { 0, 0, 0, 0 }
-  };
-int reftet_2ea_3v_splitelements[][5] =
-  {
-    { 1, 2, 3, 4, 21 },
-    { 0, 0, 0, 0 }
-  };
-HPREF_ELEMENT_TYPE reftet_2ea_3v_newelstypes[] =
-  {
-    HP_PYRAMID_EDGES,
-    HP_TET_1E_1VA,
-    HP_TET_1E_1VA,
-    HP_TET_0E_1V,
-    HP_PRISM_SINGEDGE,
-    HP_PRISM_SINGEDGE,
-
-    HP_TET, HP_TET, HP_TET, HP_TET, 
-    HP_PYRAMID, HP_PYRAMID, HP_PYRAMID, 
-    HP_PYRAMID, HP_PYRAMID, HP_TET,
-    HP_PYRAMID, HP_PYRAMID, HP_TET,
-    //     HP_PRISM,
-    //    HP_PRISM,
-    HP_NONE,
-  };
-int reftet_2ea_3v_newels[][8] =
-{
-  { 1, 5, 17, 6, 7 },
-  { 2, 8, 10, 9 },
-  { 3, 11, 12, 13 },
-  { 4, 15, 14, 16 }, 
-  { 5, 17, 7, 8, 9, 10 },
-  { 6, 7, 17, 11, 13, 12 },
- 
-  { 9, 10, 18, 21 },
-  { 13, 12, 19, 21 },
-  { 15, 16, 20, 21 },
-  { 18, 20, 19, 21 },
-  { 10, 15, 20, 18, 21 },
-  { 13, 19, 20, 16, 21 },
-  { 9, 18, 19, 12, 21 },
-  
-  { 7, 13, 16, 14, 21 },
-  { 7, 14, 15, 10, 21 },
-  { 9, 12, 17, 21 },
-  { 7, 10, 9, 17, 21 },
-  { 7, 17, 12, 13, 21 },
-  { 14, 16, 15, 21 },
-  //  { 17, 9, 12, 7, 10, 13 },
-  //  { 7, 10, 13, 14, 15, 16 },
-};
-HPRef_Struct reftet_2ea_3v =
-{
-  HP_TET,
-  reftet_2ea_3v_splitedges, 
-  reftet_2ea_3v_splitfaces, 
-  reftet_2ea_3v_splitelements, 
-  reftet_2ea_3v_newelstypes, 
-  reftet_2ea_3v_newels
-};
-
-
-
-
-
-
-//  HP_TET_2EB_4V,  // 2 opposite edges
-int reftet_2eb_4v_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 1, 8 },
-  { 2, 3, 9 },
-  { 2, 4, 10 },
-  { 3, 1, 11 },
-  { 3, 2, 12 },
-  { 3, 4, 13 },
-  { 4, 1, 14 },
-  { 4, 2, 15 },
-  { 4, 3, 16 },
-  { 0, 0, 0 }
-};
-
-HPREF_ELEMENT_TYPE reftet_2eb_4v_newelstypes[] =
-  {
-    HP_PRISM_SINGEDGE,
-    HP_PRISM_SINGEDGE,
-    HP_TET_1E_1VA,
-    HP_TET_1E_1VA,
-    HP_TET_1E_1VA,
-    HP_TET_1E_1VA,
-    HP_HEX,
-    HP_NONE,
-  };
-int reftet_2eb_4v_newels[][8] =
-{
-  { 5, 6, 7, 8, 9, 10 },
-  { 16, 15, 14, 13, 12, 11 },
-  { 1, 5, 6, 7 },
-  { 2, 8, 10, 9 },
-  { 3, 13, 11, 12 },
-  { 4, 16, 15, 14 },
-  { 7, 14, 15, 10, 6, 11, 12, 9 }
-};
-HPRef_Struct reftet_2eb_4v =
-{
-  HP_TET,
-  reftet_2eb_4v_splitedges, 
-  0, 0,
-  reftet_2eb_4v_newelstypes, 
-  reftet_2eb_4v_newels
-};
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-//  HP_TET_3EA_0V,  
-int reftet_3ea_0v_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 3, 8 },
-  { 2, 4, 9 },
-  { 3, 2, 10 },
-  { 3, 4, 11 },
-  { 4, 2, 12 },
-  { 4, 3, 13 },
-  { 0, 0, 0 }
-};
-int reftet_3ea_0v_splitfaces[][4] =
-  {
-    { 1, 2, 3, 14 },
-    { 1, 2, 4, 15 },
-    { 1, 3, 4, 16 },
-    { 2, 3, 4, 17 },
-    { 3, 4, 2, 18 },
-    { 4, 2, 3, 19 },
-    { 0, 0, 0, 0 }
-  };
-int reftet_3ea_0v_splitelements[][5] =
-  {
-    { 1, 2, 3, 4, 20 },
-    { 0 },
-  };
-
-HPREF_ELEMENT_TYPE reftet_3ea_0v_newelstypes[] =
-  {
-    HP_HEX_3E_0V,
-    HP_HEX_1E_0V,
-    HP_HEX_1E_0V,
-    HP_HEX_1E_0V,
-    HP_PRISM,
-    HP_PRISM,
-    HP_PRISM,
-    HP_TET,
-    HP_NONE,
-  };
-int reftet_3ea_0v_newels[][8] =
-{
-  { 1, 5, 14, 6, 7, 15, 20, 16 },
-  { 5, 2, 8, 14, 15, 9, 17, 20 },
-  { 3, 6, 14, 10, 11, 16, 20, 18 },
-  { 7, 4, 12, 15, 16, 13, 19, 20 },
-  { 11, 13, 16, 18, 19, 20 },
-  { 15, 12, 9, 20, 19, 17 },
-  { 8, 10, 14, 17, 18, 20 },
-  { 20, 17, 18, 19 },
-};
-HPRef_Struct reftet_3ea_0v =
-{
-  HP_TET,
-  reftet_3ea_0v_splitedges, 
-  reftet_3ea_0v_splitfaces, 
-  reftet_3ea_0v_splitelements, 
-  reftet_3ea_0v_newelstypes, 
-  reftet_3ea_0v_newels
-};
-
-
-
-
-
-
-
-//  HP_TET_3EA_3V,  
-int reftet_3ea_3v_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 3, 8 },
-  { 2, 4, 9 },
-  { 3, 2, 10 },
-  { 3, 4, 11 },
-  { 4, 2, 12 },
-  { 4, 3, 13 },
-  { 2, 1, 21 },
-  { 3, 1, 22 },
-  { 4, 1, 23 },
-  { 0, 0, 0 }
-};
-int reftet_3ea_3v_splitfaces[][4] =
-  {
-    { 1, 2, 3, 14 },
-    { 1, 2, 4, 15 },
-    { 1, 3, 4, 16 },
-    { 2, 3, 4, 17 },
-    { 3, 4, 2, 18 },
-    { 4, 2, 3, 19 },
-    { 0, 0, 0, 0 }
-  };
-int reftet_3ea_3v_splitelements[][5] =
-  {
-    { 1, 2, 3, 4, 20 },
-    { 0 },
-  };
-
-HPREF_ELEMENT_TYPE reftet_3ea_3v_newelstypes[] =
-  {
-    HP_HEX_3E_0V,
-    HP_TET_1E_1VA,
-    HP_PRISM_SINGEDGE,
-    HP_PRISM,
-    HP_TET_1E_1VA,
-    HP_PRISM_SINGEDGE,
-    HP_PRISM,
-    HP_TET_1E_1VA,
-    HP_PRISM_SINGEDGE,
-    HP_PRISM,
-
-    HP_PRISM,
-    HP_PRISM,
-    HP_PRISM,
-    HP_TET,
-    HP_NONE,
-  };
-int reftet_3ea_3v_newels[][8] =
-{
-  { 1, 5, 14, 6, 7, 15, 20, 16 },
-
-  { 2, 21, 9, 8 },
-  { 5, 14, 15, 21, 8, 9 },
-  { 15, 14, 20, 9, 8, 17 },
-  { 3, 22, 10, 11 },
-  { 6, 16, 14, 22, 11, 10 },
-  { 14, 16, 20, 10, 11, 18 },
-  { 4, 23, 13, 12 },
-  { 7, 15, 16, 23, 12, 13 },
-  { 16, 15, 20, 13, 12, 19 },
-
-  { 11, 13, 16, 18, 19, 20 },
-  { 15, 12, 9, 20, 19, 17 },
-  { 8, 10, 14, 17, 18, 20 },
-  { 20, 17, 18, 19 },
-};
-HPRef_Struct reftet_3ea_3v =
-{
-  HP_TET,
-  reftet_3ea_3v_splitedges, 
-  reftet_3ea_3v_splitfaces, 
-  reftet_3ea_3v_splitelements, 
-  reftet_3ea_3v_newelstypes, 
-  reftet_3ea_3v_newels
-};
-
-
-
-
-
-
-
-//  HP_TET_3EV_2V,  
-int reftet_3eb_2v_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 1, 8 },
-  { 2, 3, 9 },
-  { 2, 4, 10 },
-  { 3, 1, 11 },
-  { 3, 2, 12 },
-  { 3, 4, 13 },
-  { 4, 1, 14 },
-  { 4, 2, 15 },
-  { 4, 3, 16 },
-  { 0, 0, 0 }
-};
-int reftet_3eb_2v_splitfaces[][4] =
-  {
-    { 1, 2, 4, 17 },
-    { 2, 1, 3, 18 },
-    { 0, 0, 0, 0 }
-  };
-int reftet_3eb_2v_splitelements[][5] =
-  {
-    { 1, 2, 3, 4, 20 },
-    { 0 },
-  };
-
-HPREF_ELEMENT_TYPE reftet_3eb_2v_newelstypes[] =
-  {
-    HP_PYRAMID_EDGES,
-    HP_PYRAMID_EDGES,
-    HP_TET_1E_1VA,
-    HP_TET_1E_1VA,
-    HP_PRISM_SINGEDGE,
-    HP_PRISM_SINGEDGE,
-    HP_PRISM_SINGEDGE,
-    
-    HP_PYRAMID,
-    HP_PYRAMID,
-    HP_TET,
-    HP_TET,
-    HP_PYRAMID,
-    HP_PYRAMID,
-    HP_PYRAMID,
-    HP_NONE,
-  };
-int reftet_3eb_2v_newels[][8] =
-{
-  { 1, 7, 17, 5, 6 },
-  { 2, 9, 18, 8, 10 },
-  { 3, 12, 13, 11 },
-  { 4, 14, 16, 15 },
-  { 5, 6, 17, 8, 18, 10 },
-  { 7, 17, 6, 14, 15, 16 },
-  { 9, 18, 10, 12, 11, 13 },
-  
-  { 10, 15, 16, 13, 20 },
-  { 6, 11, 13, 16, 20 },
-  { 10, 17, 15, 20 },
-  { 6, 18, 11, 20 },
-  { 6, 17, 10, 18, 20 },
-  { 6, 16, 15, 17, 20 },
-  { 18, 10, 13, 11, 20 },
-};
-HPRef_Struct reftet_3eb_2v =
-{
-  HP_TET,
-  reftet_3eb_2v_splitedges, 
-  reftet_3eb_2v_splitfaces, 
-  reftet_3eb_2v_splitelements, 
-  reftet_3eb_2v_newelstypes, 
-  reftet_3eb_2v_newels
-};
-
-
-
-
-
-
-
-
-
-//  HP_TET_3EC_2V,  
-int reftet_3ec_2v_splitedges[][3] =
-{
-  { 1, 2, 5 },
-  { 1, 3, 6 },
-  { 1, 4, 7 },
-  { 2, 1, 8 },
-  { 2, 3, 9 },
-  { 2, 4, 10 },
-  { 3, 1, 11 },
-  { 3, 2, 12 },
-  { 3, 4, 13 },
-  { 4, 1, 14 },
-  { 4, 2, 15 },
-  { 4, 3, 16 },
-  { 0, 0, 0 }
-};
-int reftet_3ec_2v_splitfaces[][4] =
-  {
-    { 1, 2, 3, 17 },
-    { 2, 1, 4, 18 },
-    { 0, 0, 0, 0 }
-  };
-int reftet_3ec_2v_splitelements[][5] =
-  {
-    { 1, 2, 3, 4, 20 },
-    { 0 },
-  };
-
-HPREF_ELEMENT_TYPE reftet_3ec_2v_newelstypes[] =
-  {
-    HP_PYRAMID_EDGES,
-    HP_PYRAMID_EDGES,
-    HP_TET_1E_1VA,
-    HP_TET_1E_1VA,
-    HP_PRISM_SINGEDGE,
-    HP_PRISM_SINGEDGE,
-    HP_PRISM_SINGEDGE,
-    
-    HP_PYRAMID,
-    HP_PYRAMID,
-    HP_TET,
-    HP_TET,
-    HP_PYRAMID,
-    HP_PYRAMID,
-    HP_PYRAMID,
-    HP_NONE,
-  };
-int reftet_3ec_2v_newels[][8] =
-{
-  { 1, 5, 17, 6, 7 },
-  { 2, 8, 18, 10, 9 },
-  { 3, 11, 12, 13 },
-  { 4, 15, 14, 16 },
-  { 5, 17, 7, 8, 9, 18 },
-  { 6, 7, 17, 11, 13, 12 },
-  { 10, 9, 18, 15, 16, 14 },
-  
-  { 9, 16, 13, 12, 20 },
-  { 7, 13, 16, 14, 20 },
-  { 7, 14, 18, 20 },
-  { 9, 12, 17, 20 },
-  { 17, 7, 18, 9, 20 },
-  { 7, 17, 12, 13, 20 },
-  { 9, 18, 14, 16, 20 },
-};
-HPRef_Struct reftet_3ec_2v =
-{
-  HP_TET,
-  reftet_3ec_2v_splitedges, 
-  reftet_3ec_2v_splitfaces, 
-  reftet_3ec_2v_splitelements, 
-  reftet_3ec_2v_newelstypes, 
-  reftet_3ec_2v_newels
-};
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-// HP_PRISM
-int refprism_splitedges[][3] =
-{
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE refprism_newelstypes[] =
-{
-  HP_PRISM,
-  HP_NONE,
-};
-int refprism_newels[][8] =
-{
-  { 1, 2, 3, 4, 5, 6 }
-};
-HPRef_Struct refprism =
-{
-  HP_PRISM,
-  refprism_splitedges, 
-  0, 0,
-  refprism_newelstypes, 
-  refprism_newels
-};
-
-
-
-// HP_PRISM_SINGEDGE
-int refprism_singedge_splitedges[][3] =
-{
-  { 1, 2, 7 },
-  { 1, 3, 8 },
-  { 4, 5, 9 },
-  { 4, 6, 10 },
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE refprism_singedge_newelstypes[] =
-{
-  HP_PRISM_SINGEDGE,
-  HP_HEX,
-  HP_NONE,
-};
-int refprism_singedge_newels[][8] =
-{
-  { 1, 7, 8, 4, 9, 10 },
-  { 3, 8, 7, 2, 6, 10, 9, 5 }
-};
-HPRef_Struct refprism_singedge =
-{
-  HP_PRISM,
-  refprism_singedge_splitedges, 
-  0, 0,
-  refprism_singedge_newelstypes, 
-  refprism_singedge_newels
-};
-
-
-
-
-
-
-
-// HP_PYRAMID
-int refpyramid_splitedges[][3] =
-{
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE refpyramid_newelstypes[] =
-{
-  HP_PYRAMID,
-  HP_NONE,
-};
-int refpyramid_newels[][8] =
-{
-  { 1, 2, 3, 4, 5, 6 }
-};
-HPRef_Struct refpyramid =
-{
-  HP_PYRAMID,
-  refpyramid_splitedges, 
-  0, 0,
-  refpyramid_newelstypes, 
-  refpyramid_newels
-};
-
-
-
-// HP_PYRAMID_0E_1V
-int refpyramid_0e_1v_splitedges[][3] =
-{
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE refpyramid_0e_1v_newelstypes[] =
-{
-  HP_TET_0E_1V,
-  HP_TET,
-  HP_NONE,
-};
-int refpyramid_0e_1v_newels[][8] =
-{
-  { 1, 2, 4, 5 },
-  { 2, 3, 4, 5 },
-};
-HPRef_Struct refpyramid_0e_1v =
-{
-  HP_PYRAMID,
-  refpyramid_0e_1v_splitedges, 
-  0, 0,
-  refpyramid_0e_1v_newelstypes, 
-  refpyramid_0e_1v_newels
-};
-
-
-
-// HP_PYRAMID_EDGES
-int refpyramid_edges_splitedges[][3] =
-{
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE refpyramid_edges_newelstypes[] =
-{
-  HP_TET_1E_1VA,
-  HP_TET_1E_1VA,
-  HP_NONE,
-};
-int refpyramid_edges_newels[][8] =
-{
-  { 1, 2, 3, 5 },
-  { 1, 4, 5, 3 },
-};
-HPRef_Struct refpyramid_edges =
-{
-  HP_PYRAMID,
-  refpyramid_edges_splitedges, 
-  0, 0,
-  refpyramid_edges_newelstypes, 
-  refpyramid_edges_newels
-};
-
-
-
-
-
-
-
-// HP_HEX
-int refhex_splitedges[][3] =
-{
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE refhex_newelstypes[] =
-{
-  HP_HEX,
-  HP_NONE,
-};
-int refhex_newels[][8] =
-{
-  { 1, 2, 3, 4, 5, 6, 7, 8 }
-};
-HPRef_Struct refhex =
-{
-  HP_HEX,
-  refhex_splitedges, 
-  0, 0,
-  refhex_newelstypes, 
-  refhex_newels
-};
-
-
-
-
-// HP_HEX_0E_1V
-int refhex_0e_1v_splitedges[][3] =
-{
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE refhex_0e_1v_newelstypes[] =
-{
-  HP_TET_0E_1V,
-  HP_TET,
-  HP_TET,
-  HP_TET,
-  HP_TET,
-  HP_TET,
-  HP_NONE,
-};
-int refhex_0e_1v_newels[][8] =
-{
-  { 1, 5, 2, 4 },
-  { 7, 3, 6, 8 },
-  { 2, 8, 5, 6 },
-  { 2, 8, 6, 3 },
-  { 2, 8, 3, 4 },
-  { 2, 8, 4, 5 },
-};
-HPRef_Struct refhex_0e_1v =
-{
-  HP_HEX,
-  refhex_0e_1v_splitedges, 
-  0, 0,
-  refhex_0e_1v_newelstypes, 
-  refhex_0e_1v_newels
-};
-
-
-
-
-
-
-// HP_HEX_3E_0V
-int refhex_3e_0v_splitedges[][3] =
-{
-  { 0, 0, 0 }
-};
-HPREF_ELEMENT_TYPE refhex_3e_0v_newelstypes[] =
-{
-  HP_TET_1E_1VA,
-  HP_TET_1E_1VA,
-  HP_TET_1E_1VA,
-  HP_TET_0E_1V,
-  HP_TET,
-  HP_NONE,
-};
-int refhex_3e_0v_newels[][8] =
-{
-  { 1, 2, 3, 6 },
-  { 1, 4, 8, 3 },
-  { 1, 5, 6, 8 },
-  { 1, 6, 3, 8 },
-  { 3, 8, 6, 7 },
-};
-HPRef_Struct refhex_3e_0v =
-{
-  HP_HEX,
-  refhex_3e_0v_splitedges, 
-  0, 0,
-  refhex_3e_0v_newelstypes, 
-  refhex_3e_0v_newels
-};
-
-
-
-
-// HP_HEX_1E_0V
-int refhex_1e_0v_splitedges[][3] =
-{
-  { 0, 0, 0 }
-};
-
-HPREF_ELEMENT_TYPE refhex_1e_0v_newelstypes[] =
-{
-  HP_PRISM_SINGEDGE,
-  HP_PRISM,
-  HP_NONE,
-};
-int refhex_1e_0v_newels[][8] =
-{
-  { 1, 4, 5, 2, 3, 6 },
-  { 5, 4, 8, 6, 3, 7 },
-};
-HPRef_Struct refhex_1e_0v =
-{
-  HP_HEX,
-  refhex_1e_0v_splitedges, 
-  0, 0,
-  refhex_1e_0v_newelstypes, 
-  refhex_1e_0v_newels
-};
-
-
-
-
-
-
-
-
-HPRef_Struct * Get_HPRef_Struct (HPREF_ELEMENT_TYPE type)
-{
-  HPRef_Struct * hps = NULL;
-  switch (type)
-    {
-    case HP_TRIG:
-      hps = &reftrig; break;
-    case HP_TRIG_SINGCORNER:
-      hps = &reftrig_singcorner; break;
-    case HP_TRIG_SINGCORNER12:
-      hps = &reftrig_singcorner12; break;
-    case HP_TRIG_SINGCORNER123:
-      hps = &reftrig_singcorner123; break;
-    case HP_TRIG_SINGEDGE:
-      hps = &reftrig_singedge; break;
-    case HP_TRIG_SINGEDGECORNER1:
-      hps = &reftrig_singedgecorner1; break;
-    case HP_TRIG_SINGEDGECORNER2:
-      hps = &reftrig_singedgecorner2; break;
-    case HP_TRIG_SINGEDGECORNER12:
-      hps = &reftrig_singedgecorner12; break;
-    case HP_TRIG_SINGEDGECORNER3:
-      hps = &reftrig_singedgecorner3; break;
-    case HP_TRIG_SINGEDGECORNER13:
-      hps = &reftrig_singedgecorner13; break;
-    case HP_TRIG_SINGEDGECORNER23:
-      hps = &reftrig_singedgecorner23; break;
-    case HP_TRIG_SINGEDGECORNER123:
-      hps = &reftrig_singedgecorner123; break;
-    case HP_TRIG_SINGEDGES:
-      hps = &reftrig_singedges; break;
-    case HP_TRIG_SINGEDGES2:
-      hps = &reftrig_singedges2; break;
-    case HP_TRIG_SINGEDGES3:
-      hps = &reftrig_singedges3; break;
-    case HP_TRIG_SINGEDGES23:
-      hps = &reftrig_singedges23; break;
-    case HP_QUAD:
-      hps = &refquad; break;
-    case HP_DUMMY_QUAD_SINGCORNER:
-      hps = &refdummyquad_singcorner; break;
-    case HP_QUAD_SINGCORNER:
-      hps = &refquad_singcorner; break;
-    case HP_QUAD_SINGEDGE:
-      hps = &refquad_singedge; break;
-    case HP_QUAD_SINGEDGES:
-      hps = &refquad_singedges; break;
-
-    case HP_TET:
-      hps = &reftet; break;
-    case HP_TET_0E_1V:
-      hps = &reftet_0e_1v; break;
-    case HP_TET_0E_2V:
-      hps = &reftet_0e_2v; break;
-    case HP_TET_0E_3V:
-      hps = &reftet_0e_3v; break;
-    case HP_TET_0E_4V:
-      hps = &reftet_0e_4v; break;
-
-    case HP_TET_1E_0V:      
-      hps = &reftet_1e_0v; break;
-    case HP_TET_1E_1VA:
-      hps = &reftet_1e_1va; break;
-    case HP_TET_1E_1VB:
-      hps = &reftet_1e_1vb; break;
-
-    case HP_TET_1E_2VA:
-      hps = &reftet_1e_2va; break;
-    case HP_TET_1E_2VB:
-      hps = &reftet_1e_2vb; break;
-    case HP_TET_1E_2VC:
-      hps = &reftet_1e_2vc; break;
-    case HP_TET_1E_2VD:
-      hps = &reftet_1e_2vd; break;
-
-    case HP_TET_2EA_0V:
-      hps = &reftet_2ea_0v; break;
-    case HP_TET_2EA_1VB:
-      hps = &reftet_2ea_1vb; break;
-    case HP_TET_2EA_3V:
-      hps = &reftet_2ea_3v; break;
-
-    case HP_TET_2EB_4V:
-      hps = &reftet_2eb_4v; break;
-
-
-    case HP_TET_3EA_0V:
-      hps = &reftet_3ea_0v; break;
-    case HP_TET_3EA_3V:
-      hps = &reftet_3ea_3v; break;
-
-    case HP_TET_3EB_2V:
-      hps = &reftet_3eb_2v; break;
-    case HP_TET_3EC_2V:
-      hps = &reftet_3ec_2v; break;
-
-    case HP_PRISM:
-      hps = &refprism; break;
-    case HP_PRISM_SINGEDGE:
-      hps = &refprism_singedge; break;
-
-    case HP_PYRAMID:
-      hps = &refpyramid; break;
-    case HP_PYRAMID_0E_1V:
-      hps = &refpyramid_0e_1v; break;
-    case HP_PYRAMID_EDGES:
-      hps = &refpyramid_edges; break;
-
-    case HP_HEX:
-      hps = &refhex; break;
-    case HP_HEX_0E_1V:
-      hps = &refhex_0e_1v; break;
-    case HP_HEX_1E_0V:
-      hps = &refhex_1e_0v; break;
-    case HP_HEX_3E_0V:
-      hps = &refhex_3e_0v; break;
-    }
-
-  return hps;
-}
-
-
-
-class HPRefElement
-{
-public:
-  HPREF_ELEMENT_TYPE type;
-  int pnums[8];
-  int index;
-  int level;
-};
-
-static ARRAY<HPRefElement> elements;
-
-void PrepareElements (Mesh & mesh)
-{
-  cout << "Transform mesh to hp-elements" << endl;
-
-  int i, j, k, pi3, pi4;
-  INDEX_2_HASHTABLE<int> edges(mesh.GetNSeg()+1);
-  BitArray edgepoint(mesh.GetNP());
-  edgepoint.Clear();
-  BitArray cornerpoint(mesh.GetNP());
-  cornerpoint.Clear();
-
-  // check, if point has as least 3 different surfs:
-  ARRAY<INDEX_3> surfonpoint(mesh.GetNP());
-
-  if (mesh.GetDimension() == 3)
-    {
-      for (i = 1; i <= mesh.GetNP(); i++)
-	surfonpoint.Elem(i) = INDEX_3(0,0,0);
-      for (i = 1; i <= mesh.GetNSE(); i++)
-	{
-	  const Element2d & el = mesh.SurfaceElement(i);
-	  int ind = el.GetIndex();
-	  for (j = 0; j < el.GetNP(); j++)
-	    {
-	      int pi = el[j];
-	      INDEX_3 & i3 = surfonpoint.Elem(pi);
-	      if (ind != i3.I1() &&
-		  ind != i3.I2() && 
-		  ind != i3.I3())
-		{
-		  i3.I1() = i3.I2();
-		  i3.I2() = i3.I3();
-		  i3.I3() = ind;
-		}
-	    }
-	}
-      for (i = 1; i <= mesh.GetNP(); i++)
-	if (surfonpoint.Get(i).I1())
-	  cornerpoint.Set(i);
-      
-      // cornerpoint.Clear();
-      //  cornerpoint.Set(1);
-
-      for (i = 1; i <= mesh.GetNSeg(); i++)
-	{
-	  INDEX_2 i2 (mesh.LineSegment(i).p1, 
-		      mesh.LineSegment(i).p2);
-	  i2.Sort();
-	  // if (i2.I1() == 11 && i2.I2() == 12)
-	  {
-	    edges.Set (i2, 1);
-	    edgepoint.Set (i2.I1());
-	    edgepoint.Set (i2.I2());
-	  }
-	}
-    }
-  else
-    {
-      for (i = 1; i <= mesh.GetNP(); i++)
-	surfonpoint.Elem(i) = INDEX_3(0,0,0);
-      
-      for (i = 1; i <= mesh.GetNSeg(); i++)
-	{
-	  const Segment & seg = mesh.LineSegment(i);
-	  int ind = seg.edgenr;
-	  if (ind <= 12) continue;
-
-	  INDEX_2 i2 (mesh.LineSegment(i).p1, 
-		      mesh.LineSegment(i).p2);
-	  i2.Sort();
-	  edges.Set (i2, 1);
-	  edgepoint.Set(i2.I1());
-	  edgepoint.Set(i2.I2());
-
-	  // (*testout) << "seg = " << ind << ", " << seg.p1 << "-" << seg.p2 << endl;
-	  for (j = 0; j < 2; j++)
-	    {
-	      int pi = (j == 0) ? seg.p1 : seg.p2;
-	      // if (pi > 20)
-		{
-		  INDEX_3 & i3 = surfonpoint.Elem(pi);
-		  if (ind != i3.I1() &&
-		      ind != i3.I2())
-		    {
-		      i3.I1() = i3.I2();
-		      i3.I2() = ind;
-		    }
-		}
-	    }
-	}
-      for (i = 1; i <= mesh.GetNP(); i++)
-	if (surfonpoint.Get(i).I1())
-	  {
-	    cornerpoint.Set(i);
-	  }
-    }
-
-  int cnt_undef = 0, cnt_nonimplement = 0;
-  ARRAY<int> misses(10000);
-  misses = 0;
-
-  for (i = 1; i <= mesh.GetNE(); i++)
-    {
-      Element & el = mesh.VolumeElement(i);
-      
-      HPREF_ELEMENT_TYPE type = HP_NONE;
-      int pnums[8] = { 0 };
-
-
-      switch (el.GetType())
-	{
-	  case TET:
-	    {
-	      int ep1, ep2, ep3, ep4, cp1, cp2, cp3, cp4;
-	      int isedge1, isedge2, isedge3, isedge4, isedge5, isedge6;
-	      
-	      for (j = 1; j <= 4; j++)
-		for (k = 1; k <= 4; k++)
-		  {
-		    if (j == k) continue;
-		    if (type) break;
-	    
-	    int pi3 = 1;
-	    while (pi3 == j || pi3 == k) pi3++;
-	    pi4 = 10 - j - k - pi3;
-
-	    // preserve orientation
-	    int sort[4];
-	    sort[0] = j; sort[1] = k; sort[2] = pi3; sort[3] = pi4;
-	    int cnt = 0;
-	    for (int jj = 0; jj < 4; jj++)
-	      for (int kk = 0; kk < 3; kk++)
-		if (sort[kk] > sort[kk+1])
-		  {
-		    cnt++;
-		    Swap (sort[kk], sort[kk+1]);
-		  }
-	    if (cnt % 2 == 1) Swap (pi3, pi4);
-
-	    ep1 = edgepoint.Test (el.PNum (j));
-	    ep2 = edgepoint.Test (el.PNum (k));
-	    ep3 = edgepoint.Test (el.PNum (pi3));
-	    ep4 = edgepoint.Test (el.PNum (pi4));
-
-	    cp1 = cornerpoint.Test (el.PNum (j));
-	    cp2 = cornerpoint.Test (el.PNum (k));
-	    cp3 = cornerpoint.Test (el.PNum (pi3));
-	    cp4 = cornerpoint.Test (el.PNum (pi4));
-	    
-	    INDEX_2 i2;
-	    i2 = INDEX_2(el.PNum (j), el.PNum (k));
-	    i2.Sort();
-	    isedge1 = edges.Used (i2);
-
-	    i2 = INDEX_2(el.PNum (j), el.PNum (pi3));
-	    i2.Sort();
-	    isedge2 = edges.Used (i2);
-
-	    i2 = INDEX_2(el.PNum (j), el.PNum (pi4));
-	    i2.Sort();
-	    isedge3 = edges.Used (i2);
-
-	    i2 = INDEX_2(el.PNum (k), el.PNum (pi3));
-	    i2.Sort();
-	    isedge4 = edges.Used (i2);
-
-	    i2 = INDEX_2(el.PNum (k), el.PNum (pi4));
-	    i2.Sort();
-	    isedge5 = edges.Used (i2);
-
-	    i2 = INDEX_2(el.PNum (pi3), el.PNum (pi4));
-	    i2.Sort();
-	    isedge6 = edges.Used (i2);
-
-	    switch (isedge1+isedge2+isedge3+isedge4+isedge5+isedge6)
-	      {
-	      case 0:
-		{		
-		  if (!ep1 && !ep2 && !ep3 && !ep4)
-		    type = HP_TET;
-		  
-		  if (ep1 && !ep2 && !ep3 && !ep4)
-		    type = HP_TET_0E_1V;
-
-		  if (ep1 && ep2 && !ep3 && !ep4)
-		    type = HP_TET_0E_2V;
-
-		  if (ep1 && ep2 && ep3 && !ep4)
-		    type = HP_TET_0E_3V;
-
-		  if (ep1 && ep2 && ep3 && ep4)
-		    type = HP_TET_0E_4V;
-
-		  break;
-		}
-		
-	      case 1:
-		{
-		  if (!isedge1) break;
-		  
-		  if (!cp1 && !cp2 && !ep3 && !ep4)
-		    type = HP_TET_1E_0V;
-		    
-		  if (cp1 && !cp2 && !ep3 && !ep4)
-		    type = HP_TET_1E_1VA;
-
-		  if (!cp1 && !cp2 && !ep3 && ep4)
-		    type = HP_TET_1E_1VB;
-
-		  if (cp1 && cp2 && !ep3 && !ep4)
-		    type = HP_TET_1E_2VA;
-
-		  if (cp1 && !cp2 && ep3 && !ep4)
-		    type = HP_TET_1E_2VB;
-
-		  if (cp1 && !cp2 && !ep3 && ep4)
-		    type = HP_TET_1E_2VC;
-
-		  if (!cp1 && !cp2 && ep3 && ep4)
-		    type = HP_TET_1E_2VD;
-
-		  if (cp1 && cp2 && ep3 && !ep4)
-		    type = HP_TET_1E_3VA;
-
-		  if (cp1 && !cp2 && ep3 && ep4)
-		    type = HP_TET_1E_3VB;
-
-		  if (cp1 && cp2 && ep3 && ep4)
-		    type = HP_TET_1E_4V;
-		  
-		  break;
-		}
-
-	      case 2:
-		{
-		  if (isedge1 && isedge2)
-		    {
-		      if (!cp2 && !cp3 && !ep4)
-			type = HP_TET_2EA_0V;
-		      if (cp2 && !cp3 && !ep4)
-			type = HP_TET_2EA_1VA;
-
-		      if (!cp2 && cp3 && !ep4)
-			type = HP_TET_2EA_1VB;
-
-		      if (!cp2 && !cp3 && ep4)
-			type = HP_TET_2EA_1VC;
-
-		      if (cp2 && cp3 && !ep4)
-			type = HP_TET_2EA_2VA;
-		      if (cp2 && !cp3 && ep4)
-			type = HP_TET_2EA_2VB;
-		      if (!cp2 && cp3 && ep4)
-			type = HP_TET_2EA_2VC;
-
-		      if (cp2 && cp3 && ep4)
-			type = HP_TET_2EA_3V;
-		    }
-
-		  if (isedge1 && isedge6)
-		    {
-		      if (!cp1 && !cp2 && !cp3 && !cp4)
-			type = HP_TET_2EB_0V;
-
-		      if (cp1 && !cp2 && !cp3 && !cp4)
-			type = HP_TET_2EB_1V;
-
-		      if (cp1 && cp2 && !cp3 && !cp4)
-			type = HP_TET_2EB_2VA;
-		      if (cp1 && !cp2 && cp3 && !cp4)
-			type = HP_TET_2EB_2VB;
-		      if (cp1 && !cp2 && !cp3 && cp4)
-			type = HP_TET_2EB_2VC;
-
-		      if (cp1 && cp2 && cp3 && !cp4)
-			type = HP_TET_2EB_3V;
-
-		      if (cp1 && cp2 && cp3 && cp4)
-			type = HP_TET_2EB_4V;
-		    }
-		}
-
-	      case 3:
-		{
-		  if (isedge1 && isedge2 && isedge3)
-		    {
-		      if (!cp2 && !cp3 && !cp4)
-			type = HP_TET_3EA_0V;
-		      if (cp2 && cp3 && ep4)
-			type = HP_TET_3EA_3V;
-		    }
-		  if (isedge1 && isedge3 && isedge4)
-		    {
-		      if (cp3 && ep4)
-			type = HP_TET_3EB_2V;
-		    }
-		  if (isedge1 && isedge2 && isedge5)
-		    {
-		      if (cp3 && ep4)
-			type = HP_TET_3EC_2V;
-		    }
-		}
+    for (int i = 0; i < hpelements.Size(); i++)
+      {
+	HPRefElement & hpel = hpelements[i];
+	if (Get_HPRef_Struct (hpel.type))
+	  switch (Get_HPRef_Struct (hpel.type) -> geom)
+	    {
+	    case HP_SEGM:
+	      {
+		Segment seg;
+		seg.p1 = hpel.pnums[0];
+		seg.p2 = hpel.pnums[1];
+		// NOTE: only for less than 10000 elements (HACK) !!!
+		seg.edgenr = hpel.index % 10000;
+		seg.si     = hpel.index / 10000;
+		seg.epgeominfo[0].dist = hpel.param[0][0]; 
+		seg.epgeominfo[1].dist = hpel.param[1][0];
+		seg.epgeominfo[0].edgenr = seg.edgenr;
+		seg.epgeominfo[1].edgenr = seg.edgenr;
+		seg.hp_elnr = i;
+		mesh.AddSegment (seg);
+		break;
 	      }
+	    
 
-	    if (type != HP_NONE)
+	    case HP_TRIG:
 	      {
-		pnums[0] = el.PNumMod (j);
-		pnums[1] = el.PNumMod (k);
-		pnums[2] = el.PNumMod (pi3);
-		pnums[3] = el.PNumMod (pi4);
+		Element2d el(3);
+		el.PNum(1) = hpel.pnums[0];
+		el.PNum(2) = hpel.pnums[1];
+		el.PNum(3) = hpel.pnums[2];
+		el.SetIndex (hpel.index);
+		el.hp_elnr = i;
+		mesh.AddSurfaceElement (el);
 		break;
 	      }
-		  }
-
-
-      if (type == HP_NONE)
-	{
-	  cnt_undef++;
-	  (*testout) << "undefined element" << endl
-		     << "cp = " << cp1 << cp2 << cp3 << cp4 << endl
-		     << "ep = " << ep1 << ep2 << ep3 << ep4 << endl
-		     << "isedge = " << isedge1 << isedge2 << isedge3 
-		     << isedge4 << isedge5 << isedge6 << endl;
-	}
-      //      cout << "hpref - element = " << type << endl;
-      
-
-
-	      break;
-	    }
-	case PRISM:
-	  {
-	    int pi1, pi2, pi3, pi4, pi5, pi6;
-	    int ep1, ep2, ep3, ep4, ep5, ep6, cp1, cp2, cp3, cp4, cp5, cp6;
-
-	    int ishedge1, ishedge2, ishedge3, ishedge4, ishedge5, ishedge6;
-	    int isvedge1, isvedge2, isvedge3;
-	    
-	    for (j = 1; j <= 3; j++)
+	    case HP_QUAD:
 	      {
-		if (type) break;
-
-		pi1 = j;
-		pi2 = pi1%3 + 1;
-		pi3 = pi2%3 + 1;
-		pi4 = pi1+3;
-		pi5 = pi2+3;
-		pi6 = pi3+3;
-
-		ep1 = edgepoint.Test (el.PNum (pi1));
-		ep2 = edgepoint.Test (el.PNum (pi2));
-		ep3 = edgepoint.Test (el.PNum (pi3));
-		ep4 = edgepoint.Test (el.PNum (pi4));
-		ep5 = edgepoint.Test (el.PNum (pi5));
-		ep6 = edgepoint.Test (el.PNum (pi6));
-
-		cp1 = cornerpoint.Test (el.PNum (pi1));
-		cp2 = cornerpoint.Test (el.PNum (pi2));
-		cp3 = cornerpoint.Test (el.PNum (pi3));
-		cp4 = cornerpoint.Test (el.PNum (pi4));
-		cp5 = cornerpoint.Test (el.PNum (pi5));
-		cp6 = cornerpoint.Test (el.PNum (pi6));
-	    
-		INDEX_2 i2;
-		i2 = INDEX_2(el.PNum (pi1), el.PNum (pi4));
-		i2.Sort();
-		isvedge1 = edges.Used (i2);
-
-
-		if (isvedge1 + isvedge2 + isvedge3 == 0)
-		  {
-		    type = HP_PRISM;
-		  }
-		else if (isvedge1)
-		  {
-		    type = HP_PRISM_SINGEDGE;
-		  }
-
-
-		if (type != HP_NONE)
-		  {
-		    pnums[0] = el.PNum (pi1);
-		    pnums[1] = el.PNum (pi2);
-		    pnums[2] = el.PNum (pi3);
-		    pnums[3] = el.PNum (pi4);
-		    pnums[4] = el.PNum (pi5);
-		    pnums[5] = el.PNum (pi6);
-		    break;
-		  }
+		Element2d el(4);
+		el.PNum(1) = hpel.pnums[0];
+		el.PNum(2) = hpel.pnums[1];
+		el.PNum(3) = hpel.pnums[2];
+		el.PNum(4) = hpel.pnums[3];
+		el.SetIndex (hpel.index);
+		el.hp_elnr = i;
+		mesh.AddSurfaceElement (el);
+		break;
 	      }
-	    break;
-	  }
-	default:
-	  {
-	    cerr << "hp-refinement not defined for element" << endl;
-	  }
-	}
-
-      if (!Get_HPRef_Struct (type)) 
-	{
-	  (*testout) << "case " << type << " not implemented " << endl;
-	  cnt_nonimplement++;
-	  misses[type]++;
-	}
-      
-      HPRefElement hpel;
-      hpel.type = type;
-      for (j = 0; j < 8; j++)
-	hpel.pnums[j] = pnums[j];
-      hpel.index = el.GetIndex();
-      hpel.level = 1;
-      elements.Append (hpel);
-    }
-
-  cout << "undefined elements: " << cnt_undef << endl;
-  cout << "non-implemented: " << cnt_nonimplement << endl;
-
-  for (i = 0; i < misses.Size(); i++)
-    if (misses[i])
-      cout << "missing case " << i << " occured " << misses[i] << " times" << endl;
-
-  for (i = 1; i <= mesh.GetNSE(); i++)
-    {
-      Element2d & el = mesh.SurfaceElement(i);
-      
-      HPREF_ELEMENT_TYPE type = HP_NONE;
-      int pnums[8] = { 0 };
-
-      switch (el.GetType())
-	{
-	case TRIG:
-	  {
-	    for (j = 1; j <= 3; j++)
+	    case HP_TET:
 	      {
-		int ep1 = edgepoint.Test (el.PNumMod (j));
-		int ep2 = edgepoint.Test (el.PNumMod (j+1));
-		int ep3 = edgepoint.Test (el.PNumMod (j+2));
-		
-		int cp1 = cornerpoint.Test (el.PNumMod (j));
-		int cp2 = cornerpoint.Test (el.PNumMod (j+1));
-		int cp3 = cornerpoint.Test (el.PNumMod (j+2));
-		
-		INDEX_2 i2;
-		i2 = INDEX_2(el.PNumMod (j), el.PNumMod (j+1));
-		i2.Sort();
-		int isedge1 = edges.Used (i2);
-		i2 = INDEX_2(el.PNumMod (j+1), el.PNumMod (j+2));
-		i2.Sort();
-		int isedge2 = edges.Used (i2);
-		i2 = INDEX_2(el.PNumMod (j+2), el.PNumMod (j+3));
-		i2.Sort();
-		int isedge3 = edges.Used (i2);
-		
-		if (isedge1 + isedge2 + isedge3 == 0)
-		  {
-		    if (!ep1 && !ep2 && !ep3)
-		      type = HP_TRIG;
-		    
-		    if (ep1 && !ep2 && !ep3)
-		      type = HP_TRIG_SINGCORNER;
-		    
-		    if (ep1 && ep2 && !ep3)
-		      type = HP_TRIG_SINGCORNER12;
-		    
-		    if (ep1 && ep2 && ep3)
-		      type = HP_TRIG_SINGCORNER123;
-		    
-		    if (type != HP_NONE)
-		      {
-			pnums[0] = el.PNumMod (j);
-			pnums[1] = el.PNumMod (j+1);
-			pnums[2] = el.PNumMod (j+2);
-			break;
-		      }
-		  }
-		
-		if (isedge1 && !isedge2 && !isedge3)
-		  {
-		    int code = 0;
-		    if (cp1) code += 1;
-		    if (cp2) code += 2;
-		    if (ep3) code += 4;
-		    
-		    HPREF_ELEMENT_TYPE types[] =
-		      {
-			HP_TRIG_SINGEDGE, 
-			HP_TRIG_SINGEDGECORNER1, 
-			HP_TRIG_SINGEDGECORNER2,
-			HP_TRIG_SINGEDGECORNER12, 
-			HP_TRIG_SINGEDGECORNER3, 
-			HP_TRIG_SINGEDGECORNER13, 
-			HP_TRIG_SINGEDGECORNER23, 
-			HP_TRIG_SINGEDGECORNER123, 
-		      };
-		    type = types[code];
-		    pnums[0] = el.PNumMod (j);
-		    pnums[1] = el.PNumMod (j+1);
-		    pnums[2] = el.PNumMod (j+2);
-		    break;
-		  }
-		
-		
-		if (isedge1 && !isedge2 && isedge3)
-		  {
-		    if (!cp3)
-		      {
-			if (!cp2) type = HP_TRIG_SINGEDGES;
-			else      type = HP_TRIG_SINGEDGES2;
-		      }
-		    else
-		      {
-			if (!cp2) type = HP_TRIG_SINGEDGES3;
-			else      type = HP_TRIG_SINGEDGES23;
-		      }
-		    
-		    pnums[0] = el.PNumMod (j);
-		    pnums[1] = el.PNumMod (j+1);
-		    pnums[2] = el.PNumMod (j+2);
-		    break;
-		  }
-		
-		if (isedge1 && isedge2 && isedge3)
-		  {
-		    type = HP_TRIG_3SINGEDGES;
-		    pnums[0] = el.PNumMod (j);
-		    pnums[1] = el.PNumMod (j+1);
-		    pnums[2] = el.PNumMod (j+2);
-		    break;
-		  }
+		Element el(4);
+		el.PNum(1) = hpel.pnums[0];
+		el.PNum(2) = hpel.pnums[1];
+		el.PNum(3) = hpel.pnums[2];
+		el.PNum(4) = hpel.pnums[3];
+		el.SetIndex (hpel.index);
+		el.hp_elnr = i;
+		mesh.AddVolumeElement (el);
+		break;
 	      }
-	    break;
-	  }
-	case QUAD:
-	  {
-	    for (j = 1; j <= 4; j++)
+	    case HP_PRISM:
 	      {
-		int ep1 = edgepoint.Test (el.PNumMod (j));
-		int ep2 = edgepoint.Test (el.PNumMod (j+1));
-		int ep3 = edgepoint.Test (el.PNumMod (j+2));
-		int ep4 = edgepoint.Test (el.PNumMod (j+3));
-		
-		int cp1 = cornerpoint.Test (el.PNumMod (j));
-		int cp2 = cornerpoint.Test (el.PNumMod (j+1));
-		int cp3 = cornerpoint.Test (el.PNumMod (j+2));
-		int cp4 = cornerpoint.Test (el.PNumMod (j+3));
-		
-		INDEX_2 i2;
-		i2 = INDEX_2(el.PNumMod (j), el.PNumMod (j+1));
-		i2.Sort();
-		int isedge1 = edges.Used (i2);
-		i2 = INDEX_2(el.PNumMod (j+1), el.PNumMod (j+2));
-		i2.Sort();
-		int isedge2 = edges.Used (i2);
-		i2 = INDEX_2(el.PNumMod (j+2), el.PNumMod (j+3));
-		i2.Sort();
-		int isedge3 = edges.Used (i2);
-		i2 = INDEX_2(el.PNumMod (j+3), el.PNumMod (j+4));
-		i2.Sort();
-		int isedge4 = edges.Used (i2);
-		
-	  
-		if (isedge1 + isedge2 + isedge3 + isedge4 == 0)
-		  {
-		    type = HP_QUAD;
-		  }
-		else if (isedge1)
-		  {
-		    type = HP_QUAD_SINGEDGE;
-		  }
-
-		if (type != HP_NONE)
-		  {
-		    pnums[0] = el.PNumMod (j);
-		    pnums[1] = el.PNumMod (j+1);
-		    pnums[2] = el.PNumMod (j+2);
-		    pnums[3] = el.PNumMod (j+3);
-		    break;
-		  }
+		Element el(6);
+		el.PNum(1) = hpel.pnums[0];
+		el.PNum(2) = hpel.pnums[1];
+		el.PNum(3) = hpel.pnums[2];
+		el.PNum(4) = hpel.pnums[3];
+		el.PNum(5) = hpel.pnums[4];
+		el.PNum(6) = hpel.pnums[5];
+		el.SetIndex (hpel.index);
+		el.hp_elnr = i;
+		mesh.AddVolumeElement (el);
+		break;
 	      }
-	    break;
-	  }
-	}
-	  
-      HPRefElement hpel;
-      hpel.type = type;
-      for (j = 0; j < 8; j++)
-	hpel.pnums[j] = pnums[j];
-      hpel.index = el.GetIndex();
-      hpel.level = 1;
-      
-      elements.Append (hpel);
-    }
-}
-
-
-
-void DoRefinement (Mesh & mesh)
-{
-  int i, j, k;
-
-  INDEX_2_HASHTABLE<int> newpts(elements.Size()+1);
-  INDEX_3_HASHTABLE<int> newfacepts(elements.Size()+1);
-  
-  // prepare new points
-  
-  (*testout) << "find new points" << endl;
-  int oldelsize = elements.Size();
-  for (i = 1; i <= oldelsize; i++)
-    {
-      (*testout) << "el " << i << endl;
-
-      HPRefElement & el = elements.Elem(i);
-      HPRef_Struct * hprs = Get_HPRef_Struct (el.type);
-
-      (*testout) << "type = " << el.type << endl;
-
-      
-      if (!hprs)
-	{
-	  // cout << "Refinementstruct not defined for element " << el.type << endl;
-	  continue;
-	}
-
-      j = 0;
-      while (hprs->splitedges[j][0])
-	{
-	  INDEX_2 i2(el.pnums[hprs->splitedges[j][0]-1],
-		     el.pnums[hprs->splitedges[j][1]-1]);
-	  if (!newpts.Used (i2))
-	    {
-	      Point3d np = Center (mesh.Point (i2.I1()),
-				   mesh.Point (i2.I2()));
-	      np = Center (mesh.Point (i2.I1()),np);
-	      np = Center (mesh.Point (i2.I1()),np);
-	      int npi = mesh.AddPoint (np);
-	      newpts.Set (i2, npi);
-	    }
-	  j++;
-	}
-
-
-
-      j = 0;
-      if (hprs->splitfaces)
-	while (hprs->splitfaces[j][0])
-	  {
-	    INDEX_3 i3(el.pnums[hprs->splitfaces[j][0]-1],
-		       el.pnums[hprs->splitfaces[j][1]-1],
-		       el.pnums[hprs->splitfaces[j][2]-1]);
-	    if (i3.I2() > i3.I3())
-	      Swap (i3.I2(), i3.I3());
-	    if (!newfacepts.Used (i3))
+	    
+	    case HP_PYRAMID:
 	      {
-		Point3d np = Center (mesh.Point (i3.I2()),
-				     mesh.Point (i3.I3()));
-		np = Center (mesh.Point (i3.I1()),np);
-		np = Center (mesh.Point (i3.I1()),np);
-		int npi = mesh.AddPoint (np);
-		newfacepts.Set (i3, npi);
+		Element el(5);
+		el.PNum(1) = hpel.pnums[0];
+		el.PNum(2) = hpel.pnums[1];
+		el.PNum(3) = hpel.pnums[2];
+		el.PNum(4) = hpel.pnums[3];
+		el.PNum(5) = hpel.pnums[4];
+		el.SetIndex (hpel.index);
+		el.hp_elnr = i;
+		mesh.AddVolumeElement (el);
+		break;
+	      }
+	    case HP_HEX:
+	      {
+		Element el(8);
+		el.PNum(1) = hpel.pnums[0];
+		el.PNum(2) = hpel.pnums[1];
+		el.PNum(3) = hpel.pnums[2];
+		el.PNum(4) = hpel.pnums[3];
+		el.PNum(5) = hpel.pnums[4];
+		el.PNum(6) = hpel.pnums[5];
+		el.PNum(7) = hpel.pnums[6];
+		el.PNum(8) = hpel.pnums[7];
+		el.SetIndex (hpel.index);
+		el.hp_elnr = i;
+		mesh.AddVolumeElement (el);
+		break;
 	      }
-	    j++;
-	  }
-    }
-  
-
-
-  (*testout) << "generate elements" << endl;
-
-  for (i = 1; i <= oldelsize; i++)
-    {
-      (*testout) << "el " << i << endl;
-
-      HPRefElement & el = elements.Elem(i);
-      HPRef_Struct * hprs = Get_HPRef_Struct (el.type);
-      int newlevel = el.level + 1;
-
-      if (el.type == HP_TRIG ||
-	  el.type == HP_QUAD ||
-	  el.type == HP_TET ||
-	  el.type == HP_PRISM ||
-	  el.type == HP_HEX)
-	newlevel = el.level;
-
-      (*testout) << "type = " << el.type << endl;
-
-      if (!hprs)
-	{
-	  continue;
-	}
-
-      int newpnums[33];
-      for (j = 0; j < 8; j++)
-	newpnums[j] = el.pnums[j];
-
-      j = 0;
-      while (hprs->splitedges[j][0])
-	{
-	  INDEX_2 i2(el.pnums[hprs->splitedges[j][0]-1],
-		     el.pnums[hprs->splitedges[j][1]-1]);
-
-	  int npi = newpts.Get(i2);
-	  newpnums[hprs->splitedges[j][2]-1] = npi;
-	  j++;
-	}
-
-      j = 0;
-      if (hprs->splitfaces)
-	while (hprs->splitfaces[j][0])
-	  {
-	    INDEX_3 i3(el.pnums[hprs->splitfaces[j][0]-1],
-		       el.pnums[hprs->splitfaces[j][1]-1],
-		       el.pnums[hprs->splitfaces[j][2]-1]);
-	    if (i3.I2() > i3.I3())
-	      Swap (i3.I2(), i3.I3());
-	    int npi = newfacepts.Get(i3);
-	    newpnums[hprs->splitfaces[j][3]-1] = npi;
-	    j++;
-	  }
-
-
-      j = 0;
-      if (hprs->splitelements)
-	while (hprs->splitelements[j][0])
-	  {
-	    int pi1 = el.pnums[hprs->splitelements[j][0]-1];
-	    Point3d np = 
-	      Center (Center (mesh.Point (pi1),
-			      mesh.Point (el.pnums[hprs->splitelements[j][1]-1])),
-		      Center (mesh.Point (el.pnums[hprs->splitelements[j][2]-1]),
-			      mesh.Point (el.pnums[hprs->splitelements[j][3]-1])));
 	    
-	    np = Center (mesh.Point (pi1),np);
-	    int npi = mesh.AddPoint (np);
-	    newpnums[hprs->splitelements[j][4]-1] = npi;
-	    j++;
-	  }
-      (*testout) << "type = " << el.type << endl;
-      (*testout) << "newpnums = ";
-      for (k = 0; k < 10; k++)
-	(*testout) << newpnums[k] << " ";
-      (*testout) << endl;
-
-      j = 0;
-      while (hprs->neweltypes[j])
-	{
-	  HPRefElement newel;
-	  newel.type = hprs->neweltypes[j];
-	  for (k = 0; k < 8; k++)
-	    newel.pnums[k] = newpnums[hprs->newels[j][k]-1];
-
-	  if (newel.pnums[1] == newel.pnums[3])
-	    {
-	      cout << "same node numbers !!!! " << endl;
-	      (*testout) << "same node numbers !!!! " << endl;
+	    default:
+	      PrintSysError ("hpref, backconversion failed for element ", 
+			     int(Get_HPRef_Struct (hpel.type) -> geom));
 	    }
-	  (*testout) << "new el: ";
-	  for (k = 0; k < 8; k++)
-	    (*testout) << newel.pnums[k] << " ";
-	  (*testout) << endl;
-
-	  newel.index = elements.Elem(i).index;
-	  newel.level = newlevel;
-
-	  if (j == 0)
-	    elements.Elem(i) = newel;
-	  else
-	    elements.Append (newel);
-	  j++;
-	}
-      (*testout) << "el complete" << endl;
-    }
-
-  cout << "refinement done" << endl;
-}
-
-
-
-void DoRefineDummies (Mesh & mesh)
-{
-  cout << "refine dummies" << endl;
-  int i, j, k;
-
-  int oldelsize = elements.Size();
-
-  for (i = 1; i <= oldelsize; i++)
-    {
-      HPRefElement & el = elements.Elem(i);
-      HPRef_Struct * hprs = Get_HPRef_Struct (el.type);
-
-      int newlevel = el.level;
-
-      if (el.type != HP_DUMMY_QUAD_SINGCORNER &&
-	  el.type != HP_PYRAMID_EDGES &&
-	  el.type != HP_PYRAMID_0E_1V &&
-	  el.type != HP_HEX_0E_1V &&
-	  el.type != HP_HEX_1E_0V &&
-	  el.type != HP_HEX_3E_0V
-	  ) continue;
-
-      if (!hprs) continue;
-
-      int newpnums[33];
-      for (j = 0; j < 8; j++)
-	newpnums[j] = el.pnums[j];
-
-      j = 0;
-      while (hprs->neweltypes[j])
-	{
-	  HPRefElement newel;
-	  newel.type = hprs->neweltypes[j];
-	  for (k = 0; k < 8; k++)
-	    newel.pnums[k] = newpnums[hprs->newels[j][k]-1];
-	  newel.index = el.index;
-
-	  newel.level = newlevel;
-
-	  if (j == 0)
-	    elements.Elem(i) = newel;
-	  else
-	    elements.Append (newel);
-	  j++;
-	}
-    }
-  cout << "refineme dummies done" << endl;
-}
-
-
-
-
-
-
-
-void CalcStatistics ()
-{
-  int i, p;
-  int ntrig = 0, nquad = 0;
-  int nhex = 0, nprism = 0, ntet = 0;
-  int maxlevel = 0;
-
-  for (i = 1; i <= elements.Size(); i++)
-    {
-      const HPRefElement & el = elements.Get(i);
-      maxlevel = max2 (el.level, maxlevel);
-      switch (el.type)
-	{
-	case HP_TRIG:
-	case HP_TRIG_SINGCORNER:
-	case HP_TRIG_SINGEDGE:
-	case HP_TRIG_SINGEDGECORNER1:
-	case HP_TRIG_SINGEDGECORNER2:
-	  {
-	    ntrig ++;
-	    break;
-	  }
-	case HP_QUAD:
-	case HP_QUAD_SINGEDGE:
-	  {
-	    nquad++;
-	    break;
-	  }
-	case HP_TET:
-	case HP_TET_0E_1V:
-	case HP_TET_1E_0V:
-	case HP_TET_1E_1VA:
-	  {
-	    ntet++;
-	    break;
-	  }
-
-	case HP_PRISM:
-	case HP_PRISM_SINGEDGE:
-	  {
-	    nprism++;
-	    break;
-	  }
-
-	case HP_HEX:
-	  {	
-	    nhex++;
-	    break;
-	  }
-	}
-    }
-
-  cout << "level = " << maxlevel << endl;
-  cout << "ntrig = " << ntrig << ", nquad = " << nquad << endl;
-  cout << "ntet = " << ntet << ", nprism = " << nprism << ", nhex = " << nhex << endl;
-
-  return;
-
-  double memcost = 0, cpucost = 0;
-  for (p = 1; p <= 20; p++)
-    {
-      memcost = (ntet + nprism + nhex) * pow (p, 6.0);
-      cpucost = (ntet + nprism + nhex) * pow (p, 9.0);
-      cout << "costs for p = " << p << ": mem = " << memcost << ", cpu = " << cpucost << endl;
-    }
-
-  double memcosttet = 0;
-  double memcostprism = 0;
-  double memcosthex = 0;
-  double memcostsctet = 0;
-  double memcostscprism = 0;
-  double memcostschex = 0;
-  double cpucosttet = 0;
-  double cpucostprism = 0;
-  double cpucosthex = 0;
-
-  for (i = 1; i <= elements.Size(); i++)
-    {
-      const HPRefElement & el = elements.Get(i);
-      switch (el.type)
-	{
-	case HP_TET:
-	case HP_TET_0E_1V:
-	case HP_TET_1E_0V:
-	case HP_TET_1E_1VA:
-	  {
-	    int p1 = maxlevel - el.level + 1;
-	    (*testout) << "p1 = " << p1 << ", P1^6 = " << pow (p1, 6.0)
-		       << " (p1-3)^6 = " << pow ( max2(p1-3, 0), 6.0) 
-		       << " p1^3 = " << pow ( p1, 3.0) 
-		       << " (p1-3)^3 = " << pow ( p1-3, 3.0) 
-		       << " [p1^3-(p1-3)^3]^2 = " << sqr (pow (p1,3.0) - pow ( p1-3, 3.0))
-		       << endl;
-
-	    p1 /= 2 +1;
-	    memcosttet += pow (p1, 6.0);
-	    memcostsctet += pow (p1, 6.0) - pow ( max2(p1-3, 1), 6.0);
-	    cpucosttet += pow (p1, 9.0);
-	    break;
-	  }
-	case HP_PRISM:
-	case HP_PRISM_SINGEDGE:
-	  {
-	    int p1 = maxlevel - el.level + 1;
-	    p1 /= 2 +1;
-	    memcostprism += pow (p1, 6.0);
-	    memcostscprism += pow (p1, 6.0) - pow ( max2(p1-3, 1), 6.0);
-	    cpucostprism += pow (p1, 9.0);
-	    break;
-	  }
-	case HP_HEX:
-	  {	
-	    int p1 = maxlevel - el.level + 1;
-	    int p2 = maxlevel;
-	    p1 /= 2 +1;
-	    p2 /= 2 +1;
-	    memcosthex += pow (p1, 4.0) * pow (p2, 2.0);
-	    memcostschex += pow (p1, 6.0) - pow ( max2(p1-2, 0), 6.0);
-	    cpucosthex += pow (p1, 6.0) * pow (p2, 3.0);
-	    break;
-	  }
-	default:
-	  ;
-	}
-    }
-  cout << "TET: hp-memcost = " << memcosttet 
-       << ", scmemcost = " << memcostsctet
-       << ", cpucost = " << cpucosttet
-       << endl;
-  cout << "PRI: hp-memcost = " << memcostprism
-       << ", scmemcost = " << memcostscprism
-       << ", cpucost = " << cpucostprism << endl;
-  cout << "HEX: hp-memcost = " << memcosthex
-       << ", scmemcost = " << memcostschex
-       << ", cpucost = " << cpucosthex << endl;
-}
-
-void HPRefinement (Mesh & mesh, int levels)
-{
-  int i, j;
-  cout << "HP Refinement called, levels = " << levels;
-
-  PrepareElements (mesh);
-
-  for (j = 1; j <= levels; j++)
-    {
-      DoRefinement (mesh);
-      DoRefineDummies (mesh);
-      CalcStatistics ();
-    }
-  
-  mesh.ClearSurfaceElements();
-  mesh.ClearVolumeElements();
-
-  for (i = 1; i <= elements.Size(); i++)
-    {
-      HPRefElement & hpel = elements.Elem(i);
-      if (Get_HPRef_Struct (hpel.type))
-	switch (Get_HPRef_Struct (hpel.type) -> geom)
-	{
-	case HP_TRIG:
-	  {
-	    Element2d el(3);
-	    el.PNum(1) = hpel.pnums[0];
-	    el.PNum(2) = hpel.pnums[1];
-	    el.PNum(3) = hpel.pnums[2];
-	    el.SetIndex (hpel.index);
-	    mesh.AddSurfaceElement (el);
-	    break;
-	  }
-	case HP_QUAD:
-	  {
-	    Element2d el(4);
-	    el.PNum(1) = hpel.pnums[0];
-	    el.PNum(2) = hpel.pnums[1];
-	    el.PNum(3) = hpel.pnums[2];
-	    el.PNum(4) = hpel.pnums[3];
-	    el.SetIndex (hpel.index);
-	    mesh.AddSurfaceElement (el);
-	    break;
-	  }
-	case HP_TET:
-	  {
-	    Element el(4);
-	    el.PNum(1) = hpel.pnums[0];
-	    el.PNum(2) = hpel.pnums[1];
-	    el.PNum(3) = hpel.pnums[2];
-	    el.PNum(4) = hpel.pnums[3];
-	    el.SetIndex (hpel.index);
-	    mesh.AddVolumeElement (el);
-	    break;
-	  }
-	case HP_PRISM:
-	  {
-	    Element el(6);
-	    el.PNum(1) = hpel.pnums[0];
-	    el.PNum(2) = hpel.pnums[1];
-	    el.PNum(3) = hpel.pnums[2];
-	    el.PNum(4) = hpel.pnums[3];
-	    el.PNum(5) = hpel.pnums[4];
-	    el.PNum(6) = hpel.pnums[5];
-	    el.SetIndex (hpel.index);
-	    mesh.AddVolumeElement (el);
-	    break;
-	  }
-
-	case HP_PYRAMID:
-	  {
-	    Element el(5);
-	    el.PNum(1) = hpel.pnums[0];
-	    el.PNum(2) = hpel.pnums[1];
-	    el.PNum(3) = hpel.pnums[2];
-	    el.PNum(4) = hpel.pnums[3];
-	    el.PNum(5) = hpel.pnums[4];
-	    el.SetIndex (hpel.index);
-	    mesh.AddVolumeElement (el);
-	    break;
-	  }
-	case HP_HEX:
-	  {
-	    Element el(8);
-	    el.PNum(1) = hpel.pnums[0];
-	    el.PNum(2) = hpel.pnums[1];
-	    el.PNum(3) = hpel.pnums[2];
-	    el.PNum(4) = hpel.pnums[3];
-	    el.PNum(5) = hpel.pnums[4];
-	    el.PNum(6) = hpel.pnums[5];
-	    el.PNum(7) = hpel.pnums[6];
-	    el.PNum(8) = hpel.pnums[7];
-	    
-	    (*testout) << "hex: " << endl;
-	    for (int ii = 1;  ii <= 8; ii++)
-	      (*testout) << hpel.pnums[ii-1] << " ";
-	    (*testout) << endl;
-
-	    el.SetIndex (hpel.index);
-	    mesh.AddVolumeElement (el);
-	    break;
-	  }
-
-	default:
-	  PrintSysError ("hpref, backconversion failed for element ", 
-			 int(Get_HPRef_Struct (hpel.type) -> geom));
-	}
-    }
+      }
   
-  mesh.UpdateTopology();
-}
+    mesh.UpdateTopology();
+  }
 
 
 }
diff --git a/Netgen/libsrc/meshing/hprefinement.hpp b/Netgen/libsrc/meshing/hprefinement.hpp
index ca21a886d6..4537b41345 100644
--- a/Netgen/libsrc/meshing/hprefinement.hpp
+++ b/Netgen/libsrc/meshing/hprefinement.hpp
@@ -12,7 +12,212 @@
 */
 
 
-extern void HPRefinement (Mesh & mesh, int levels);
+
+
+enum HPREF_ELEMENT_TYPE {
+  HP_NONE=0,
+
+  HP_SEGM = 1,
+  HP_SEGM_SINGCORNERL,
+  HP_SEGM_SINGCORNERR,
+  HP_SEGM_SINGCORNERS,
+
+  HP_TRIG = 10,
+  HP_TRIG_SINGCORNER,
+  HP_TRIG_SINGCORNER12,
+  HP_TRIG_SINGCORNER123,
+  HP_TRIG_SINGCORNER123_2D,   // not rotational symmetric
+  HP_TRIG_SINGEDGE = 20,
+  HP_TRIG_SINGEDGECORNER1,   // E = 100, V = 100
+  HP_TRIG_SINGEDGECORNER2,   // E = 100, V = 010
+  HP_TRIG_SINGEDGECORNER12,  // E = 100, V = 110
+  HP_TRIG_SINGEDGECORNER3,
+  HP_TRIG_SINGEDGECORNER13,
+  HP_TRIG_SINGEDGECORNER23,
+  HP_TRIG_SINGEDGECORNER123,
+  HP_TRIG_SINGEDGES = 30,
+  HP_TRIG_SINGEDGES2,
+  HP_TRIG_SINGEDGES3,
+  HP_TRIG_SINGEDGES23,
+  HP_TRIG_3SINGEDGES = 40,
+
+  HP_QUAD = 50,
+  HP_QUAD_SINGCORNER,
+  HP_DUMMY_QUAD_SINGCORNER,
+  HP_QUAD_SINGEDGE,
+  HP_QUAD_0E_2VA,  // V = 1100
+  HP_QUAD_0E_2VB,  // V = 1010
+  HP_QUAD_0E_3V,
+  HP_QUAD_0E_4V,
+
+  // one edge: marked edge is always edge from vertex 1 to vertex 2 (E = 1000)
+  HP_QUAD_1E_1VA,  // vertex on beginning of edge: V = 1000
+  HP_QUAD_1E_1VB,  // vertex on end of edge: V = 0100
+  HP_QUAD_1E_1VC,  // V = 0010
+  HP_QUAD_1E_1VD,  // V = 0001
+
+  HP_QUAD_1E_2VA,  // V = 1100
+  HP_QUAD_1E_2VB,  // V = 1010
+  HP_QUAD_1E_2VC,  // V = 1001
+  HP_QUAD_1E_2VD,  // V = 0110
+  HP_QUAD_1E_2VE,  // V = 0101
+  HP_QUAD_1E_2VF,  // V = 0011
+
+  HP_QUAD_1E_3VA,  // V = 1110
+  HP_QUAD_1E_3VB,  // V = 1101
+  HP_QUAD_1E_3VC,  // V = 1011
+  HP_QUAD_1E_3VD,  // V = 0111
+
+  HP_QUAD_1E_4V,   // V = 1111
+
+
+  HP_QUAD_2E,      // E = 1001, V = 1000
+  HP_QUAD_2E_1VA,  // E = 1001, V = 1100
+  HP_QUAD_2E_1VB,  // E = 1001, V = 1010
+  HP_QUAD_2E_1VC,  // E = 1001, V = 1001
+  HP_QUAD_2E_2VA,  // E = 1001, V = 1110
+  HP_QUAD_2E_2VB,  // E = 1001, V = 1101
+  HP_QUAD_2E_2VC,  // E = 1001, V = 1011
+  HP_QUAD_2E_3V,   // E = 1001, V = 1111
+
+  HP_QUAD_2EB_0V,   // E = 1010, V = 0000
+  HP_QUAD_2EB_1VA,  // E = 1010, V = 1000
+  HP_QUAD_2EB_1VB,  // E = 1010, V = 0100
+  HP_QUAD_2EB_2VA,  // E = 1010, V = 1100
+  HP_QUAD_2EB_2VB,  // E = 1010, V = 1010
+  HP_QUAD_2EB_2VC,  // E = 1010, V = 1001
+  HP_QUAD_2EB_2VD,  // E = 1010, V = 0101
+  HP_QUAD_2EB_3VA,  // E = 1010, V = 1110
+  HP_QUAD_2EB_3VB,  // E = 1010, V = 1101
+
+  HP_QUAD_2EB_4V,
+
+
+  HP_QUAD_3E,      // E = 1101, V = 1100
+  HP_QUAD_3E_3VA,  // E = 1101, V = 1110
+  HP_QUAD_3E_3VB,  // E = 1101, V = 1101
+  HP_QUAD_3E_4V,   // E = 1101, V = 1111
+
+  HP_QUAD_4E,
+
+
+  HP_TET = 100,     // no singular vertex/edge
+  HP_TET_0E_1V,     // V1
+  HP_TET_0E_2V,     // V1,2
+  HP_TET_0E_3V,     // V1,2,3  
+  HP_TET_0E_4V,     // V1,2,3,4
+  HP_TET_1E_0V = 200,   // E1-2
+  HP_TET_1E_1VA,    // V1
+  HP_TET_1E_1VB,    // V3
+  HP_TET_1E_2VA,    // V1,2
+  HP_TET_1E_2VB,    // V1,3
+  HP_TET_1E_2VC,    // V1,4
+  HP_TET_1E_2VD,    // V3,4
+  HP_TET_1E_3VA,    // V1,2,3
+  HP_TET_1E_3VB,    // V1,3,4
+  HP_TET_1E_4V,     // V1,2,3,4
+
+
+  // 2 connected edges, additonally marked Vs
+  HP_TET_2EA_0V = 220,    // E1-2, E1-3
+  HP_TET_2EA_1VA,   // V2
+  HP_TET_2EA_1VB,   // V3
+  HP_TET_2EA_1VC,   // V4
+  HP_TET_2EA_2VA,   // V2,3
+  HP_TET_2EA_2VB,   // V2,4
+  HP_TET_2EA_2VC,   // V3,4
+  HP_TET_2EA_3V,    // V2,3,4
+
+  // 2 opposite edges
+  HP_TET_2EB_0V = 230,    // E1-2, E3-4
+  HP_TET_2EB_1V,    // V1
+  HP_TET_2EB_2VA,   // V1,2
+  HP_TET_2EB_2VB,   // V1,3
+  HP_TET_2EB_2VC,   // V1,4
+  HP_TET_2EB_3V,    // V1,2,3
+  HP_TET_2EB_4V,    // V1,2,3,4
+
+  HP_TET_3EA_0V = 400,  // E1-2, E1-3, E1-4, 3 edges connected
+  HP_TET_3EA_1V,        // V2
+  HP_TET_3EA_2V,        // V2,3
+  HP_TET_3EA_3V,        // V2,3,4
+
+  HP_TET_3EB_0V = 420,  // E1-2, E1-4, E2-3  3 edges chain
+  HP_TET_3EB_1V,        // 
+  HP_TET_3EB_2V,        // 
+  HP_TET_3EC_0V = 430,  // 3 edges chain, alter
+  HP_TET_3EC_1V,        // 3 edges chain, alter
+  HP_TET_3EC_2V,        // 3 edges chain, alter
+
+
+  HP_TET_1F_0E_0V = 500,  // 1 singular face
+  HP_TET_1F_0E_1VA,       // 1 sing vertex in face (V2)
+  HP_TET_1F_0E_1VB,       // 1 sing vertex not in face (V1)
+  HP_TET_1F_1EA_0V,       // 1 sing edge not in face
+  HP_TET_1F_1EB_0V,       // 1 sing edge in face
+  HP_TET_2F_0E_0V = 600,  // 2 singular faces
+
+  HP_PRISM = 1000,
+  HP_PRISM_SINGEDGE,
+  HP_PRISM_SINGEDGE_V12,
+  HP_PRISM_SINGEDGE_H1,
+  HP_PRISM_SINGEDGE_H12,
+
+  HP_PRISM_1FA_0E_0V,     // 1 singular trig face
+  HP_PRISM_1FB_0E_0V,     // 1 singular quad face  1-2-4-5
+  HP_PRISM_1FB_1EA_0V,     // 1 singular quad face, edge is 1-2
+
+  HP_PYRAMID = 2000,
+  HP_PYRAMID_0E_1V,
+  HP_PYRAMID_EDGES,
+  HP_PYRAMID_1FB_0E_1VA,  // 1 trig face, top vertex
+
+  HP_HEX = 3000,
+  HP_HEX_0E_1V,
+  HP_HEX_1E_1V,
+  HP_HEX_1E_0V,
+  HP_HEX_3E_0V,
+
+  HP_HEX_1F_0E_0V,
+};
+
+
+
+struct HPRef_Struct {
+  HPREF_ELEMENT_TYPE geom;
+  int (*splitedges)[3];
+  int (*splitfaces)[4];
+  int (*splitelements)[5];
+  HPREF_ELEMENT_TYPE * neweltypes;
+  int (*newels)[8];
+};
+
+
+
+
+class HPRefElement
+{
+public:
+  HPRefElement () 
+  {
+    for (int i = 0; i < 8; i++)
+      {
+	pnums[i] = -1;
+	param[i][0] = param[i][1] = param[i][2] = 0;
+      }
+  }
+  HPREF_ELEMENT_TYPE type;
+  PointIndex pnums[8];
+  double param[8][3];
+  int index;
+  int level;
+  int coarse_elnr;
+  //  EdgePointGeomInfo epgeominfo[2];
+};
+
+
+
+extern void HPRefinement (Mesh & mesh, Refinement * ref, int levels);
 
 
 #endif
diff --git a/Netgen/libsrc/meshing/improve2.cpp b/Netgen/libsrc/meshing/improve2.cpp
index e70944d69b..af0ea0ca45 100644
--- a/Netgen/libsrc/meshing/improve2.cpp
+++ b/Netgen/libsrc/meshing/improve2.cpp
@@ -3,9 +3,9 @@
 #include "meshing.hpp"
 #include <opti.hpp>
 
-//removed for gmsh
+#ifndef SMALLLIB
 //#include <visual.hpp>
-
+#endif
 
 namespace netgen
 {
@@ -44,6 +44,8 @@ public:
  
 void MeshOptimize2d :: EdgeSwapping (Mesh & mesh, int usemetric)
 {
+  //  return; 
+
   if (!faceindex)
     {
       if (usemetric)
@@ -60,9 +62,11 @@ void MeshOptimize2d :: EdgeSwapping (Mesh & mesh, int usemetric)
 	}
 
       faceindex = 0;
+      mesh.CalcSurfacesOfNode();
       return;
     }
 
+
   int i, i2, j, k, j2;
   bool should;
   PointIndex pi;
@@ -247,21 +251,25 @@ void MeshOptimize2d :: EdgeSwapping (Mesh & mesh, int usemetric)
 	      gi3 = mesh[t1].GeomInfoPiMod(o1);
 	      gi4 = mesh[t2].GeomInfoPiMod(o2);
 	      
-	      // normal of old
+	      // normal of old   (new ?????)
 	      nv1 = Cross (mesh.Point(pi3)-mesh.Point(pi4), 
 			   mesh.Point(pi1)-mesh.Point(pi4));
 	      nv2 = Cross (mesh.Point(pi4)-mesh.Point(pi3), 
 			   mesh.Point(pi2)-mesh.Point(pi3));
 
 	      
-	      // normals of swapped
+	      // normals of swapped  original (???JS)
 	      Vec3d nv3, nv4;
 	      nv3 = Cross (mesh.Point(pi1)-mesh.Point(pi4), 
 			   mesh.Point(pi2)-mesh.Point(pi4));
 	      nv4 = Cross (mesh.Point(pi2)-mesh.Point(pi3), 
 			   mesh.Point(pi1)-mesh.Point(pi3));
 	      
-	      
+	      nv3 *= -1;
+	      nv4 *= -1;
+	      nv3.Normalize();
+	      nv4.Normalize();
+
 	      nv1.Normalize();
 	      nv2.Normalize();
 	    
@@ -287,8 +295,9 @@ void MeshOptimize2d :: EdgeSwapping (Mesh & mesh, int usemetric)
 		(nvp3 * nv3 > critval) && 
 		(nvp4 * nv4 > critval);
 	      
+
 	      horder = Dist (mesh.Point(pi1), mesh.Point(pi2));
-	      
+
 	      if ( // nv1 * nv2 >= 0 &&
 		  nv1.Length() > 1e-3 * horder * horder &&
 		  nv2.Length() > 1e-3 * horder * horder &&
@@ -314,13 +323,14 @@ void MeshOptimize2d :: EdgeSwapping (Mesh & mesh, int usemetric)
 					     metricweight, loch) +
 			CalcTriangleBadness (mesh.Point(pi2), mesh.Point(pi1), mesh.Point(pi4), 
 					     metricweight, loch);
+
 		    }
 
 		  
 		  if (allowswap)
 		    {
 		      Element2d sw1 (pi4, pi3, pi1);
-		      Element2d sw2 (pi4, pi4, pi2);
+		      Element2d sw2 (pi3, pi4, pi2);
 
 		      int legal1 = 
 			mesh.LegalTrig (mesh.SurfaceElement (t1)) + 
@@ -336,6 +346,8 @@ void MeshOptimize2d :: EdgeSwapping (Mesh & mesh, int usemetric)
 		    {
 		      // do swapping !
 		      
+		      // cout << "swap " << endl;
+
 		      nswaps ++;
 		      
 		    //            testout << "nv1 = " << nv1 << "   nv2 = " << nv2 << endl;
@@ -363,8 +375,8 @@ void MeshOptimize2d :: EdgeSwapping (Mesh & mesh, int usemetric)
 		      pdef[pi3]++;
 		      pdef[pi4]++;
 		      
-		      swapped.Elem(t1) = 1;
-		      swapped.Elem(t2) = 1;
+		      swapped[t1] = 1;
+		      swapped[t2] = 1;
 		    }
 		}
 	    }
@@ -422,7 +434,7 @@ void MeshOptimize2d :: CombineImprove (Mesh & mesh)
 
   int should;
   PointIndex pi1, pi2;
-  Point3d p1, p2, pnew;
+  MeshPoint p1, p2, pnew;
   double bad1, bad2;
   Vec3d nv;
 
@@ -436,9 +448,12 @@ void MeshOptimize2d :: CombineImprove (Mesh & mesh)
     {
       Element2d & el = mesh[seia[i]];
       for (j = 0; j < el.GetNP(); j++)
-	elementsonnode.Add (el[j], seia[i]);
+	{
+	  elementsonnode.Add (el[j], seia[i]);
+	}      
     }
 
+
   ARRAY<int,PointIndex::BASE> fixed(np);
   fixed = 0;
 
@@ -472,10 +487,11 @@ void MeshOptimize2d :: CombineImprove (Mesh & mesh)
 	    }
 	}
     }
-	
+
 
   for (i = 0; i < seia.Size(); i++)
     {
+
       sei = seia[i];
       Element2d & elem = mesh[sei];
       if (elem.IsDeleted()) continue;
@@ -508,7 +524,7 @@ void MeshOptimize2d :: CombineImprove (Mesh & mesh)
 	  // save version:
 	  if (fixed.Get(pi1) || fixed.Get(pi2)) 
 	    continue;
-	  if (pi2 < pi1) Swap (pi1, pi2);
+	  if (pi2 < pi1) swap (pi1, pi2);
 	  */
 
 	  // more general 
@@ -550,6 +566,7 @@ void MeshOptimize2d :: CombineImprove (Mesh & mesh)
 		}
 	    } 
 
+
 	  Element2d & hel = mesh[hasbothpi[0]];
 	  for (k = 0; k < 3; k++)
 	    if (hel[k] == pi1)
@@ -568,6 +585,7 @@ void MeshOptimize2d :: CombineImprove (Mesh & mesh)
 	  //	  nv = normals.Get(pi1);
 
 
+
 	  for (k = 0; k < elementsonnode[pi2].Size(); k++)
 	    {
 	      const Element2d & el2 = mesh[elementsonnode[pi2][k]];
@@ -632,6 +650,7 @@ void MeshOptimize2d :: CombineImprove (Mesh & mesh)
 	  mesh[pi1] = p1;
 	  mesh[pi2] = p2;
 	  
+       
 	  if (debugflag)
 	    {
 	      (*testout) << "bad1 = " << bad1 << ", bad2 = " << bad2 << endl;
@@ -654,17 +673,29 @@ void MeshOptimize2d :: CombineImprove (Mesh & mesh)
 
 	      mesh[pi1] = pnew;
 	      PointGeomInfo gi;
+	      bool gi_set(false);
 	      
-	      Element2d & el1 = mesh[elementsonnode[pi1][0]];
-	      for (l = 0; l < el1.GetNP(); l++)
-		if (el1[l] == pi1)
-		  gi = el1.GeomInfoPi (l+1);
-		  
+	      
+	      Element2d *el1p;
+	      l=0;
+	      while(mesh[elementsonnode[pi1][l]].IsDeleted() && l<elementsonnode.EntrySize(pi1)) l++;
+	      if(l<elementsonnode.EntrySize(pi1))
+		el1p = &mesh[elementsonnode[pi1][l]];
+	      else
+		cerr << "OOPS!" << endl;
+
+	      for (l = 0; l < el1p->GetNP(); l++)
+		if ((*el1p)[l] == pi1)
+		  {
+		    gi = el1p->GeomInfoPi (l+1);
+		    gi_set = true;
+		  }
 
 	      // (*testout) << "Connect point " << pi2 << " to " << pi1 << "\n";
 	      for (k = 0; k < elementsonnode[pi2].Size(); k++)
 		{
 		  Element2d & el = mesh[elementsonnode[pi2][k]];
+		  if(el.IsDeleted()) continue;
 		  elementsonnode.Add (pi1, elementsonnode[pi2][k]);
 
 		  bool haspi1 = 0;
@@ -703,6 +734,7 @@ void MeshOptimize2d :: CombineImprove (Mesh & mesh)
 		    mesh[hasbothpi[k]][l] = PointIndex::BASE-1;
 		  */
 		}
+
 	    }
 	}
     }
diff --git a/Netgen/libsrc/meshing/improve2.hpp b/Netgen/libsrc/meshing/improve2.hpp
index 39cd1be018..0770a93b05 100644
--- a/Netgen/libsrc/meshing/improve2.hpp
+++ b/Netgen/libsrc/meshing/improve2.hpp
@@ -41,7 +41,7 @@ public:
   virtual int CalcPointGeomInfo(PointGeomInfo& gi, const Point3d& /*p3*/) const
     { gi.trignum = 1; return 1;};
 
-  virtual int CalcPointGeomInfo(int surfind, PointGeomInfo& gi, const Point3d& p3) const
+  virtual int CalcPointGeomInfo(int /* surfind */, PointGeomInfo& gi, const Point3d& p3) const
     { return CalcPointGeomInfo (gi, p3); }
 
   ///
diff --git a/Netgen/libsrc/meshing/improve2gen.cpp b/Netgen/libsrc/meshing/improve2gen.cpp
index 40291575f5..69f0e23b27 100644
--- a/Netgen/libsrc/meshing/improve2gen.cpp
+++ b/Netgen/libsrc/meshing/improve2gen.cpp
@@ -11,6 +11,7 @@ namespace netgen
   public:
     ARRAY<Element2d> oldels;
     ARRAY<Element2d> newels;
+    ARRAY<INDEX_2> deledges;
     ARRAY<int> incelsonnode;
     ARRAY<int> reused;
     int bonus;
@@ -31,10 +32,10 @@ namespace netgen
 	faceindex = 0;
       }
 
-    int j, k, l, ri;
+    // int j, k, l, ri;
     int np = mesh.GetNP();
     int ne = mesh.GetNSE();
-    SurfaceElementIndex sei;
+    //    SurfaceElementIndex sei;
 
     bool ok;
     int olddef, newdef;
@@ -54,6 +55,7 @@ namespace netgen
     r1->oldels.Append (Element2d (1, 2, 3));
     r1->oldels.Append (Element2d (3, 2, 4));
     r1->newels.Append (Element2d (1, 2, 4, 3));
+    r1->deledges.Append (INDEX_2 (2,3));
     r1->onp = 4;
     r1->bonus = 2;
     rules.Append (r1);
@@ -63,6 +65,8 @@ namespace netgen
     r1->oldels.Append (Element2d (1, 2, 3, 4));
     r1->oldels.Append (Element2d (4, 3, 2, 5));
     r1->newels.Append (Element2d (1, 2, 5, 4));
+    r1->deledges.Append (INDEX_2 (2, 3));
+    r1->deledges.Append (INDEX_2 (3, 4));
     r1->onp = 5;
     r1->bonus = 0;
     rules.Append (r1);
@@ -73,6 +77,7 @@ namespace netgen
     r1->oldels.Append (Element2d (3, 2, 5, 6));
     r1->newels.Append (Element2d (1, 6, 3, 4));
     r1->newels.Append (Element2d (1, 2, 5, 6));
+    r1->deledges.Append (INDEX_2 (2, 3));
     r1->onp = 6;
     r1->bonus = 0;
     rules.Append (r1);
@@ -84,6 +89,9 @@ namespace netgen
     r1->oldels.Append (Element2d (3, 6, 7, 4));
     r1->newels.Append (Element2d (1, 2, 5, 4));
     r1->newels.Append (Element2d (4, 5, 6, 7));
+    r1->deledges.Append (INDEX_2 (2, 3));
+    r1->deledges.Append (INDEX_2 (3, 4));
+    r1->deledges.Append (INDEX_2 (3, 6));
     r1->onp = 7;
     r1->bonus = -1;
     rules.Append (r1);
@@ -94,6 +102,9 @@ namespace netgen
     r1->oldels.Append (Element2d (2, 5, 3));
     r1->oldels.Append (Element2d (3, 5, 4));
     r1->newels.Append (Element2d (1, 2, 5, 4));
+    r1->deledges.Append (INDEX_2 (2, 3));
+    r1->deledges.Append (INDEX_2 (3, 4));
+    r1->deledges.Append (INDEX_2 (3, 5));
     r1->onp = 5;
     r1->bonus = 0;
     rules.Append (r1);
@@ -105,6 +116,8 @@ namespace netgen
     r1->oldels.Append (Element2d (1, 4, 5));
     r1->newels.Append (Element2d (1, 3, 4, 5));
     r1->newels.Append (Element2d (1, 2, 6, 3));
+    r1->deledges.Append (INDEX_2 (1, 4));
+    r1->deledges.Append (INDEX_2 (2, 3));
     r1->onp = 6;
     r1->bonus = 0;
     rules.Append (r1);
@@ -115,6 +128,8 @@ namespace netgen
     r1->oldels.Append (Element2d (1, 4, 5));
     r1->newels.Append (Element2d (1, 2, 4, 5));
     r1->newels.Append (Element2d (4, 2, 6, 3));
+    r1->deledges.Append (INDEX_2 (1, 4));
+    r1->deledges.Append (INDEX_2 (2, 3));
     r1->onp = 6;
     r1->bonus = 0;
     rules.Append (r1);
@@ -126,6 +141,9 @@ namespace netgen
     r1->oldels.Append (Element2d (4, 3, 6));
     r1->newels.Append (Element2d (1, 2, 6, 4));
     r1->newels.Append (Element2d (2, 5, 6));
+    r1->deledges.Append (INDEX_2 (2, 3));
+    r1->deledges.Append (INDEX_2 (3, 4));
+    r1->deledges.Append (INDEX_2 (3, 6));
     r1->onp = 6;
     r1->bonus = -1;
     rules.Append (r1);
@@ -136,6 +154,7 @@ namespace netgen
     r1->oldels.Append (Element2d (2, 5, 3));
     r1->newels.Append (Element2d (2, 5, 3, 4));
     r1->newels.Append (Element2d (1, 2, 4));
+    r1->deledges.Append (INDEX_2 (2, 3));
     r1->onp = 5;
     r1->bonus = 0;
     rules.Append (r1);
@@ -145,11 +164,25 @@ namespace netgen
     r1->oldels.Append (Element2d (2, 5, 3));
     r1->newels.Append (Element2d (1, 2, 5, 3));
     r1->newels.Append (Element2d (1, 3, 4));
+    r1->deledges.Append (INDEX_2 (2, 3));
     r1->onp = 5;
     r1->bonus = 0;
     rules.Append (r1);
 
 
+    // 2 quads to quad + 2 trigs
+    r1 = new ImprovementRule;
+    r1->oldels.Append (Element2d (1, 2, 3, 4));
+    r1->oldels.Append (Element2d (3, 2, 5, 6));
+    r1->newels.Append (Element2d (1, 5, 6, 4));
+    r1->newels.Append (Element2d (1, 2, 5));
+    r1->newels.Append (Element2d (4, 6, 3));
+    r1->deledges.Append (INDEX_2 (2, 3));
+    r1->onp = 6;
+    r1->bonus = 0;
+    //    rules.Append (r1);
+
+
 
 
     ARRAY<int> mapped(rules.Size());
@@ -159,29 +192,29 @@ namespace netgen
 
   
 
-    for (ri = 0; ri < rules.Size(); ri++)
+    for (int ri = 0; ri < rules.Size(); ri++)
       {
 	ImprovementRule & rule = *rules[ri];
 	rule.incelsonnode.SetSize (rule.onp);
 	rule.reused.SetSize (rule.onp);
 
-	for (j = 1; j <= rule.onp; j++)
+	for (int j = 1; j <= rule.onp; j++)
 	  {
 	    rule.incelsonnode.Elem(j) = 0;
 	    rule.reused.Elem(j) = 0;
 	  }
 
-	for (j = 1; j <= rule.oldels.Size(); j++)
+	for (int j = 1; j <= rule.oldels.Size(); j++)
 	  {
 	    const Element2d & el = rule.oldels.Elem(j);
-	    for (k = 1; k <= el.GetNP(); k++)
+	    for (int k = 1; k <= el.GetNP(); k++)
 	      rule.incelsonnode.Elem(el.PNum(k))--;
 	  }
 
-	for (j = 1; j <= rule.newels.Size(); j++)
+	for (int j = 1; j <= rule.newels.Size(); j++)
 	  {
 	    const Element2d & el = rule.newels.Elem(j);
-	    for (k = 1; k <= el.GetNP(); k++)
+	    for (int k = 1; k <= el.GetNP(); k++)
 	      {
 		rule.incelsonnode.Elem(el.PNum(k))++;
 		rule.reused.Elem(el.PNum(k)) = 1;
@@ -194,34 +227,34 @@ namespace netgen
   
     TABLE<int,PointIndex::BASE> elonnode(np);
     ARRAY<int,PointIndex::BASE> nelonnode(np);
-    TABLE<int> nbels(ne);
+    TABLE<SurfaceElementIndex> nbels(ne);
 
     nelonnode = -4;
 
-    for (sei = 0; sei < ne; sei++)
+    for (SurfaceElementIndex sei = 0; sei < ne; sei++)
       {
 	const Element2d & el = mesh[sei];
-	if (el.GetIndex() == faceindex)
+	if (el.GetIndex() == faceindex && !el.IsDeleted())
 	  {
-	    for (j = 0; j < el.GetNP(); j++)
+	    for (int j = 0; j < el.GetNP(); j++)
 	      elonnode.Add (el[j], sei);
 	  }
-	for (j = 0; j < el.GetNP(); j++)
+	for (int j = 0; j < el.GetNP(); j++)
 	  nelonnode[el[j]]++;
       }
 
-    for (sei = 0; sei < ne; sei++)
+    for (SurfaceElementIndex sei = 0; sei < ne; sei++)
       {
 	const Element2d & el = mesh[sei];
-	if (el.GetIndex() == faceindex)
+	if (el.GetIndex() == faceindex && !el.IsDeleted())
 	  {
-	    for (j = 0; j < el.GetNP(); j++)
+	    for (int j = 0; j < el.GetNP(); j++)
 	      {
-		for (k = 0; k < elonnode[el[j]].Size(); k++)
+		for (int k = 0; k < elonnode[el[j]].Size(); k++)
 		  {
 		    int nbel = elonnode[el[j]] [k];
 		    int used = 0;
-		    for (l = 0; l < nbels[sei].Size(); l++)
+		    for (int l = 0; l < nbels[sei].Size(); l++)
 		      if (nbels[sei][l] == nbel)
 			used = 1;
 		    if (!used)
@@ -232,7 +265,7 @@ namespace netgen
       }
 
 
-    for (ri = 0; ri < rules.Size(); ri++)
+    for (int ri = 0; ri < rules.Size(); ri++)
       {
 	const ImprovementRule & rule = *rules[ri];
       
@@ -242,221 +275,164 @@ namespace netgen
 	pgi.SetSize (rule.onp);
 
 
-	// loop over elements and rotations:
-	/*
-	  int nt = int (pow (ne, elmap.Size()));
-	  for (i = 1; i <= nt; i++)
-	  {
-	  int hi = i-1;
-	  for (j = 1; j <= elmap.Size(); j++)
-	  {
-	  elmap.Elem (j) = hi % ne + 1;
-	  hi /= ne;
-	  }
-	*/
-
-	for (sei = 0; sei < ne; sei++)
+	for (SurfaceElementIndex sei = 0; sei < ne; sei++)
 	  {
 	    if (multithread.terminate)
 	      break;
+	    if (mesh[sei].IsDeleted()) continue;
 
 	    elmap[0] = sei;
+	    FlatArray<SurfaceElementIndex> neighbours = nbels[sei];
+	    
+	    for (elrot[0] = 0; elrot[0] < mesh[sei].GetNP(); elrot[0]++)
+	      {
+		const Element2d & el0 = mesh[sei];
+		const Element2d & rel0 = rule.oldels[0];
 
-	    int nnb = nbels[sei].Size();
-	    int nt = int (pow (double(nnb), double(elmap.Size()-1)));
+		if (el0.GetIndex() != faceindex) continue;
+		if (el0.IsDeleted()) continue;
+		if (el0.GetNP() != rel0.GetNP()) continue;
 
-	    for (int i = 1; i <= nt; i++)
-	      {
-		int hi = i-1;
-		for (j = 1; j < elmap.Size(); j++)
-		  {
-		    elmap[j] = nbels[sei][hi % nnb];
-		    hi /= nnb;
-		  }
 
-		ok = 1;
-		for (j = 0; j < elmap.Size(); j++)
+		pmap = -1;
+ 
+		for (int k = 0; k < el0.GetNP(); k++)
 		  {
-		    const Element2d & el = mesh.SurfaceElement(elmap[j]);
-		    const Element2d & rel = rule.oldels[j];
-		   
-		    if (el.GetNP() != rel.GetNP()) { ok = 0; break; }
-		    if (el.IsDeleted()) { ok = 0; break; }
+		    pmap.Elem(rel0[k]) = el0.PNumMod(k+elrot[0]+1);
+		    pgi.Elem(rel0[k]) = el0.GeomInfoPiMod(k+elrot[0]+1);
 		  }
 		
-		if (!ok) continue;
+		ok = 1;
+		for (int i = 1; i < elmap.Size(); i++)
+		  {
+		    // try to find a mapping for reference-element i
 
+		    const Element2d & rel = rule.oldels[i];
+		    bool possible = 0;
 
-		int nr = int (pow (4.0, elmap.Size()));
-		for (int i2 = 1; i2 <= nr; i2++)
-		  {
-		    int hi2 = i2-1;
-		    for (j = 1; j <= elmap.Size(); j++)
+		    for (elmap[i] = 0; elmap[i] < neighbours.Size(); elmap[i]++)
 		      {
-			elrot.Elem (j) = hi2 % 4 + 1;
-			hi2 /= 4;
-		      }
-		  
-		  
-		    // check applicable 
-		  
-		    ok = 1;
+			const Element2d & el = mesh[neighbours[elmap[i]]];
+			if (el.IsDeleted()) continue;
+			if (el.GetNP() != rel.GetNP()) continue;
 
-		    // set mapped points
-		    for (j = 1; j <= elmap.Size(); j++)
-		      {
-			const Element2d & el = mesh.SurfaceElement(elmap.Get(j));
-			const Element2d & rel = rule.oldels.Get(j);
-		      
-			/*
-			if (el.GetNP() != rel.GetNP())
+			for (elrot[i] = 0; elrot[i] < rel.GetNP(); elrot[i]++)
 			  {
-			    ok = 0; 
-			    break;
-			  }
-			*/
+			    possible = 1;
 
-			for (k = 1; k <= el.GetNP(); k++)
-			  {
-			    /*
-			    if (el.PNum(k) < PointIndex::BASE)
+			    for (int k = 0; k < rel.GetNP(); k++)
+			      if (pmap.Elem(rel[k]) != -1 &&
+				  pmap.Elem(rel[k]) != el.PNumMod (k+elrot[i]+1))
+				possible = 0;
+
+			    if (possible) 
 			      {
-				ok = 0;
+				for (int k = 0; k < el.GetNP(); k++)
+				  {
+				    pmap.Elem(rel[k]) = el.PNumMod(k+elrot[i]+1);
+				    pgi.Elem(rel[k]) = el.GeomInfoPiMod(k+elrot[i]+1);
+				  }
 				break;
 			      }
-			    */
-			    pmap.Elem(rel.PNum(k)) =
-			      el.PNumMod(k+elrot.Get(j));
-			    pgi.Elem(rel.PNum(k)) = 
-			      el.GeomInfoPiMod(k+elrot.Get(j));
 			  }
+			if (possible) break;
 		      }
-		    if (!ok) continue;
 
-		    /*
-		      (*testout) << "candidates found: " << endl;
-		      for (j = 1; j <= elmap.Size(); j++)
+		    if (!possible) 
 		      {
-		      const Element2d & el = mesh.SurfaceElement(elmap.Get(j));
-		      (*testout) << "el " << elmap.Get(j) 
-		      << ", rot = " << elrot.Get(j) << " "
-		      << el << endl;
+			ok = 0;
+			break;
 		      }
-		    */
-		  
-	      
-		    // check consistently mapped points:
-		    for (j = 1; j <= elmap.Size(); j++)
-		      {
-			const Element2d & el = mesh.SurfaceElement(elmap.Get(j));
-			const Element2d & rel = rule.oldels.Get(j);
-		      
-			for (k = 1; k <= el.GetNP(); k++)
-			  {
-			    if (pmap.Elem(rel.PNum(k)) != el.PNumMod(k+elrot.Get(j)))
-			      ok = 0;
-			  }
-		      }  
 
-		    if (!ok) continue;
+		    elmap[i] = neighbours[elmap[i]];
+		  }
 
+		for(int i=0; ok && i<rule.deledges.Size(); i++)
+		  {
+		    ok = !mesh.IsSegment(pmap.Elem(rule.deledges[i].I1()),
+					 pmap.Elem(rule.deledges[i].I2()));
+		  }
+								    
+								    
+		
+		
+		if (!ok) continue;
+
+		mapped[ri]++;
+
+		olddef = 0;
+		for (int j = 1; j <= pmap.Size(); j++)
+		  olddef += sqr (nelonnode[pmap.Get(j)]);
+		olddef += rule.bonus;
+
+		newdef = 0;
+		for (int j = 1; j <= pmap.Size(); j++)
+		  if (rule.reused.Get(j))
+		    newdef += sqr (nelonnode[pmap.Get(j)] + 
+				   rule.incelsonnode.Get(j));
+
+		if (newdef > olddef)
+		  continue;
+
+		// calc metric badness
+		double bad1 = 0, bad2 = 0;
+		Vec3d n;
+
+		SelectSurfaceOfPoint (mesh.Point(pmap.Get(1)), pgi.Get(1));
+		GetNormalVector (surfnr, mesh.Point(pmap.Get(1)), pgi.Elem(1), n);
 		  
-		    mapped[ri]++;
-		  
-		    /*
-		      (*testout) << "application found: " << endl;
-		      for (j = 1; j <= elmap.Size(); j++)
-		      {
-		      const Element2d & el = mesh.SurfaceElement(elmap.Get(j));
-		      (*testout) << "el " << elmap.Get(j) 
-		      << ", rot = " << elrot.Get(j) << " "
-		      << el << endl;
-		      }
-		      (*testout) << "points:";
-		      for (j = 1; j <= pmap.Size(); j++)
-		      (*testout) << " " << pmap.Get(j);
-		      (*testout) << endl;
-		    */
-
-		    olddef = 0;
-		    for (j = 1; j <= pmap.Size(); j++)
-		      olddef += sqr (nelonnode[pmap.Get(j)]);
-		    olddef += rule.bonus;
-
-		    newdef = 0;
-		    for (j = 1; j <= pmap.Size(); j++)
-		      if (rule.reused.Get(j))
-			newdef += sqr (nelonnode[pmap.Get(j)] + 
-				       rule.incelsonnode.Get(j));
-
-		    if (newdef > olddef)
-		      continue;
-
-		    // calc metric badness
-		    double bad1 = 0, bad2 = 0;
-		    Vec3d n;
-
-		    SelectSurfaceOfPoint (mesh.Point(pmap.Get(1)), pgi.Get(1));
-		    GetNormalVector (surfnr, mesh.Point(pmap.Get(1)), pgi.Elem(1), n);
-		  
-		    for (j = 1; j <= rule.oldels.Size(); j++)
-		      bad1 += mesh.SurfaceElement(elmap.Get(j)).CalcJacobianBadness (mesh.Points(), n);
+		for (int j = 1; j <= rule.oldels.Size(); j++)
+		  bad1 += mesh.SurfaceElement(elmap.Get(j)).CalcJacobianBadness (mesh.Points(), n);
 		  
-		    // check new element:
-		    for (j = 1; j <= rule.newels.Size(); j++)
-		      {
-			const Element2d & rnel = rule.newels.Get(j);
-			Element2d nel(rnel.GetNP());
-			for (k = 1; k <= rnel.GetNP(); k++)
-			  nel.PNum(k) = pmap.Get(rnel.PNum(k));
+		// check new element:
+		for (int j = 1; j <= rule.newels.Size(); j++)
+		  {
+		    const Element2d & rnel = rule.newels.Get(j);
+		    Element2d nel(rnel.GetNP());
+		    for (int k = 1; k <= rnel.GetNP(); k++)
+		      nel.PNum(k) = pmap.Get(rnel.PNum(k));
 
-			bad2 += nel.CalcJacobianBadness (mesh.Points(), n);
-		      }
+		    bad2 += nel.CalcJacobianBadness (mesh.Points(), n);
+		  }
 
-		    if (bad2 > 1e3) continue;
+		if (bad2 > 1e3) continue;
 
-		    if (newdef == olddef && bad2 > bad1) continue;
+		if (newdef == olddef && bad2 > bad1) continue;
 		  
-	  
-		    // generate new element:
-		    for (j = 1; j <= rule.newels.Size(); j++)
+
+		// generate new element:
+		for (int j = 1; j <= rule.newels.Size(); j++)
+		  {
+		    const Element2d & rnel = rule.newels.Get(j);
+		    Element2d nel(rnel.GetNP());
+		    nel.SetIndex (faceindex);
+		    for (int k = 1; k <= rnel.GetNP(); k++)
 		      {
-			const Element2d & rnel = rule.newels.Get(j);
-			Element2d nel(rnel.GetNP());
-			nel.SetIndex (faceindex);
-			for (k = 1; k <= rnel.GetNP(); k++)
-			  {
-			    nel.PNum(k) = pmap.Get(rnel.PNum(k));
-			    nel.GeomInfoPi(k) = pgi.Get(rnel.PNum(k));
-			  }
-		      
-			mesh.AddSurfaceElement(nel);
+			nel.PNum(k) = pmap.Get(rnel.PNum(k));
+			nel.GeomInfoPi(k) = pgi.Get(rnel.PNum(k));
 		      }
+		      
+		    mesh.AddSurfaceElement(nel);
+		  }
 		  
-		    for (j = 1; j <= rule.oldels.Size(); j++)
-		      mesh.DeleteSurfaceElement (elmap.Get(j));
+		for (int j = 0; j < rule.oldels.Size(); j++)
+		  mesh.DeleteSurfaceElement ( elmap[j] );
 
-		    for (j = 1; j <= pmap.Size(); j++)
-		      nelonnode[pmap.Get(j)] += rule.incelsonnode.Get(j);
+		for (int j = 1; j <= pmap.Size(); j++)
+		  nelonnode[pmap.Get(j)] += rule.incelsonnode.Get(j);
 
-		    used[ri]++;
-		    break;
-		  }
+		used[ri]++;
 	      }
 	  }
       }
 
     mesh.Compress();
 
-    for (ri = 0; ri < rules.Size(); ri++)
+    for (int ri = 0; ri < rules.Size(); ri++)
       {
 	PrintMessage (5, "rule ", ri+1, " ",
 		      mapped[ri], "/", used[ri], " mapped/used");
       }
-
-    int sum = 0;
-    for (int i = 0; i < used.Size(); i++)
-      sum += used[i];
   }
 
 
diff --git a/Netgen/libsrc/meshing/improve3.cpp b/Netgen/libsrc/meshing/improve3.cpp
index 625abaef65..1b9c8c52f6 100644
--- a/Netgen/libsrc/meshing/improve3.cpp
+++ b/Netgen/libsrc/meshing/improve3.cpp
@@ -21,22 +21,10 @@ namespace netgen
 void MeshOptimize3d :: CombineImprove (Mesh & mesh,
 				       OPTIMIZEGOAL goal)
 {
-  int j, k, l;
-
-  ElementIndex ei;
-  PointIndex pi1, pi2;
-
-  Point3d p1, p2, pnew;
-
-  double bad1, bad2;
-  
   int np = mesh.GetNP();
   int ne = mesh.GetNE();
 
   TABLE<ElementIndex, PointIndex::BASE> elementsonnode(np); 
-  ARRAY<Element*> hasonepoint;
-  ARRAY<Element*> hasbothpoints;
-
   ARRAY<ElementIndex> hasonepi, hasbothpi;
 
   ARRAY<double> oneperr;
@@ -46,35 +34,35 @@ void MeshOptimize3d :: CombineImprove (Mesh & mesh,
   (*testout)  << "Start CombineImprove" << "\n";
 
   //  mesh.CalcSurfacesOfNode ();
-
   char * savetask = multithread.task;
   multithread.task = "Combine Improve";
 
-  int cnt = 0;
 
-  bad1 = 0;
-  for (ei = 0; ei < ne; ei++)
+  double totalbad = 0;
+  for (ElementIndex ei = 0; ei < ne; ei++)
     {
       double elerr = CalcBad (mesh.Points(), mesh[ei], 0);
-      bad1 += elerr;
+      totalbad += elerr;
       elerrs[ei] = elerr;
     }
 
   if (goal == OPT_QUALITY)
     {
-      bad1 = CalcTotalBad (mesh.Points(), mesh.VolumeElements());
-      (*testout) << "Total badness = " << bad1 << endl;
-      PrintMessage (5, "Total badness = ", bad1);
+      totalbad = CalcTotalBad (mesh.Points(), mesh.VolumeElements());
+      (*testout) << "Total badness = " << totalbad << endl;
+      PrintMessage (5, "Total badness = ", totalbad);
     }
 
-  for (ei = 0; ei < ne; ei++)
-    for (j = 0; j < mesh[ei].GetNP(); j++)
-      elementsonnode.Add (mesh[ei][j], ei);
+  for (ElementIndex ei = 0; ei < ne; ei++)
+    if (!mesh[ei].IsDeleted())
+      for (int j = 0; j < mesh[ei].GetNP(); j++)
+	elementsonnode.Add (mesh[ei][j], ei);
   
-  INDEX_2_HASHTABLE<int> edgetested (np);
+  INDEX_2_HASHTABLE<int> edgetested (np+1);
 
+  int cnt = 0;
 
-  for (ei = 0; ei < ne; ei++)
+  for (ElementIndex ei = 0; ei < ne; ei++)
     {
       if (multithread.terminate)
 	break;
@@ -84,7 +72,7 @@ void MeshOptimize3d :: CombineImprove (Mesh & mesh,
       if (mesh.ElementType(ei) == FIXEDELEMENT)
 	continue;
 
-      for (j = 0; j < 6; j++)
+      for (int j = 0; j < 6; j++)
 	{
 	  Element & elemi = mesh[ei];
 	  if (elemi.IsDeleted()) continue;
@@ -93,8 +81,8 @@ void MeshOptimize3d :: CombineImprove (Mesh & mesh,
 	    { { 0, 1 }, { 0, 2 }, { 0, 3 },
 	      { 1, 2 }, { 1, 3 }, { 2, 3 } };
 
-	  pi1 = elemi[tetedges[j][0]];
-	  pi2 = elemi[tetedges[j][1]];
+	  PointIndex pi1 = elemi[tetedges[j][0]];
+	  PointIndex pi2 = elemi[tetedges[j][1]];
 
 	  if (pi2 < pi1) Swap (pi1, pi2);
 	  
@@ -105,13 +93,13 @@ void MeshOptimize3d :: CombineImprove (Mesh & mesh,
 	  edgetested.Set (si2, 1);
 
 
-	  hasonepoint.SetSize(0);
-	  hasbothpoints.SetSize(0);
+	  // hasonepoint.SetSize(0);
+	  //	  hasbothpoints.SetSize(0);
 	  hasonepi.SetSize(0);
 	  hasbothpi.SetSize(0);
 
 	  FlatArray<ElementIndex> row1 = elementsonnode[pi1];
-	  for (k = 0; k < row1.Size(); k++)
+	  for (int k = 0; k < row1.Size(); k++)
 	    {
 	      Element & elem = mesh[row1[k]];
 	      if (elem.IsDeleted()) continue;
@@ -119,18 +107,16 @@ void MeshOptimize3d :: CombineImprove (Mesh & mesh,
 	      if (elem[0] == pi2 || elem[1] == pi2 ||
 		  elem[2] == pi2 || elem[3] == pi2)
 		{
-		  hasbothpoints.Append (&elem);
 		  hasbothpi.Append (row1[k]);
 		}
 	      else
 		{
-		  hasonepoint.Append (&elem);    
 		  hasonepi.Append (row1[k]);
 		}
 	    } 
 	  
 	  FlatArray<ElementIndex> row2 = elementsonnode[pi2];	  
-	  for (k = 0; k < row2.Size(); k++)
+	  for (int k = 0; k < row2.Size(); k++)
 	    {
 	      Element & elem = mesh[row2[k]];
 	      if (elem.IsDeleted()) continue;
@@ -140,25 +126,26 @@ void MeshOptimize3d :: CombineImprove (Mesh & mesh,
 		;
 	      else
 		{
-		  hasonepoint.Append (&elem);    
 		  hasonepi.Append (row2[k]);
 		}
 	    } 
 	  
-	  bad1 = 0;
-	  for (k = 0; k < hasonepoint.Size(); k++)
+	  double bad1 = 0;
+	  for (int k = 0; k < hasonepi.Size(); k++)
 	    bad1 += elerrs[hasonepi[k]];
-	  
-	  for (k = 0; k < hasbothpoints.Size(); k++)
+	  for (int k = 0; k < hasbothpi.Size(); k++)
 	    bad1 += elerrs[hasbothpi[k]];
 	  
-	  p1 = mesh[pi1];
-	  p2 = mesh[pi2];
+	  MeshPoint p1 = mesh[pi1];
+	  MeshPoint p2 = mesh[pi2];
 	  
-	  if (mesh.PointType(pi2) != INNERPOINT)
+	  // if (mesh.PointType(pi2) != INNERPOINT)
+	  if (p2.Type() != INNERPOINT)
 	    continue;
 	  
-	  if (mesh.PointType(pi1) != INNERPOINT)
+	  MeshPoint pnew;
+	  // if (mesh.PointType(pi1) != INNERPOINT)
+	  if (p1.Type() != INNERPOINT)
 	    pnew = p1;
 	  else
 	    pnew = Center (p1, p2);
@@ -166,11 +153,12 @@ void MeshOptimize3d :: CombineImprove (Mesh & mesh,
 	  mesh[pi1] = pnew;
 	  mesh[pi2] = pnew;
 
-	  oneperr.SetSize (hasonepoint.Size());
-	  bad2 = 0;
-	  for (k = 0; k < hasonepoint.Size(); k++)
+	  oneperr.SetSize (hasonepi.Size());
+
+	  double bad2 = 0;
+	  for (int k = 0; k < hasonepi.Size(); k++)
 	    {
-	      const Element & elem = *hasonepoint[k];
+	      const Element & elem = mesh[hasonepi[k]];
 	      double err = CalcTetBadness (mesh[elem[0]], mesh[elem[1]],  
 					   mesh[elem[2]], mesh[elem[3]], 0);
 	      bad2 += err;
@@ -181,11 +169,13 @@ void MeshOptimize3d :: CombineImprove (Mesh & mesh,
 	  mesh[pi2] = p2;
 
 
-	  if (mesh.PointType(pi1) != INNERPOINT)
+	  // if (mesh.PointType(pi1) != INNERPOINT)
+	  if (p1.Type() != INNERPOINT)
 	    {
-	      for (k = 0; k < hasonepoint.Size(); k++)
+	      for (int k = 0; k < hasonepi.Size(); k++)
 		{
-		  Element & elem = *hasonepoint[k];
+		  Element & elem = mesh[hasonepi[k]];
+		  int l;
 		  for (l = 0; l < 4; l++)
 		    if (elem[l] == pi2)
 		      {
@@ -205,33 +195,35 @@ void MeshOptimize3d :: CombineImprove (Mesh & mesh,
 		}
 	    }
 
-	  if (bad2 / hasonepoint.Size()  < 
-	      bad1 / (hasonepoint.Size()+hasbothpoints.Size())  )
+	  if (bad2 / hasonepi.Size()  < 
+	      bad1 / (hasonepi.Size()+hasbothpi.Size()))
 	    {
 	      mesh[pi1] = pnew;
 	      cnt++;
 
 	      FlatArray<ElementIndex> row = elementsonnode[pi2];
-	      for (k = 0; k < row.Size(); k++)
+	      for (int k = 0; k < row.Size(); k++)
 		{
 		  Element & elem = mesh[row[k]];
+		  if (elem.IsDeleted()) continue;
+
 		  elementsonnode.Add (pi1, row[k]);
-		  for (l = 0; l < elem.GetNP(); l++)
+		  for (int l = 0; l < elem.GetNP(); l++)
 		    if (elem[l] == pi2)
 		      elem[l] = pi1;
 		  
 		  elem.flags.illegal_valid = 0;
 		  if (!mesh.LegalTet (elem))
-		    (*testout) << "illegal tet " << elementsonnode.Get (pi2, k) << endl;
+		    (*testout) << "illegal tet " << elementsonnode[pi2][k] << endl;
 		}
 
-	      for (k = 0; k < hasonepoint.Size(); k++)
+	      for (int k = 0; k < hasonepi.Size(); k++)
 		elerrs[hasonepi[k]] = oneperr[k];
 	      
-	      for (k = 0; k < hasbothpoints.Size(); k++)
+	      for (int k = 0; k < hasbothpi.Size(); k++)
 		{
-		  hasbothpoints[k] -> flags.illegal_valid = 0;
-		  hasbothpoints[k] -> Delete();
+		  mesh[hasbothpi[k]].flags.illegal_valid = 0;
+		  mesh[hasbothpi[k]].Delete();
 		}
 	    }
 	}
@@ -243,18 +235,18 @@ void MeshOptimize3d :: CombineImprove (Mesh & mesh,
   PrintMessage (5, cnt, " elements combined");
   (*testout) << "CombineImprove done" << "\n";
 
-  bad1 = 0;
-  for (ei = 0; ei < mesh.GetNE(); ei++)
-    bad1 += CalcBad (mesh.Points(), mesh[ei], 0);
+  totalbad = 0;
+  for (ElementIndex ei = 0; ei < mesh.GetNE(); ei++)
+    totalbad += CalcBad (mesh.Points(), mesh[ei], 0);
 
   if (goal == OPT_QUALITY)
     {
-      bad1 = CalcTotalBad (mesh.Points(), mesh.VolumeElements());
-      (*testout) << "Total badness = " << bad1 << endl;
+      totalbad = CalcTotalBad (mesh.Points(), mesh.VolumeElements());
+      (*testout) << "Total badness = " << totalbad << endl;
 
       int cntill = 0;
       int ne = mesh.GetNE();
-      for (ei = 0; ei < ne; ei++)
+      for (ElementIndex ei = 0; ei < ne; ei++)
 	if (!mesh.LegalTet (mesh[ei]))
 	  cntill++;
 
@@ -529,7 +521,7 @@ void MeshOptimize3d :: SplitImprove (Mesh & mesh,
 	      if (!mesh.LegalTet (newel2)) bad2 += 1e6;
 	    }	  
 	  
-	  mesh.PointTypes().DeleteLast();
+	  // mesh.PointTypes().DeleteLast();
 	  mesh.Points().DeleteLast();
 
 	  if (bad2 < bad1) 
@@ -594,12 +586,9 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 {
   int j, k, l;
 
-  (*testout) << "swapimprove" << endl;
-  
   ElementIndex ei;
   SurfaceElementIndex sei;
 
-  int mattyp;
   PointIndex pi1, pi2, pi3, pi4, pi5, pi6;
   int cnt = 0;
 
@@ -710,7 +699,10 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 	  const Element & elemi = mesh[ei];
 	  if (elemi.IsDeleted()) continue;
 
-	  mattyp = elemi.GetIndex();
+
+	  //	  (*testout) << "check element " << elemi << endl;
+
+	  int mattyp = elemi.GetIndex();
 	  
 	  static const int tetedges[6][2] =
 	    { { 0, 1 }, { 0, 2 }, { 0, 3 },
@@ -860,8 +852,8 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 		  face.Sort();
 		  if (faces.Used(face))
 		    {
-		      (*testout) << "3->2 swap, could improve conformity, bad1 = " << bad1
-				 << ", bad2 = " << bad2 << endl;
+		      // (*testout) << "3->2 swap, could improve conformity, bad1 = " << bad1
+		      //				 << ", bad2 = " << bad2 << endl;
 		      if (bad2 < 1e4)
 			bad1 = 2 * bad2;
 		    }
@@ -890,6 +882,7 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 		  cnt++;
 		  
 
+		  /*
 		  (*testout) << "3->2 swap, old els = " << endl
 			     << mesh[hasbothpoints[0]] << endl
 			     << mesh[hasbothpoints[1]] << endl
@@ -897,7 +890,7 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 			     << "new els = " << endl
 			     << el21 << endl
 			     << el22 << endl;
-
+		  */
 
 
 		  el21.flags.illegal_valid = 0;
@@ -1157,6 +1150,7 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 		  //		  (*mycout) << "bad1 = " << bad1 << " bad2 = " << bad2 << "\n";
 
 
+		  /*
 		  (*testout) << "4->4 swap A, old els = " << endl
 			     << mesh[hasbothpoints[0]] << endl
 			     << mesh[hasbothpoints[1]] << endl
@@ -1167,7 +1161,7 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 			     << el2 << endl
 			     << el3 << endl
 			     << el4 << endl;
-
+		  */
 
 
 
@@ -1194,7 +1188,7 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 		  el4b.flags.illegal_valid = 0;
 
 
-
+		  /*
 		  (*testout) << "4->4 swap A, old els = " << endl
 			     << mesh[hasbothpoints[0]] << endl
 			     << mesh[hasbothpoints[1]] << endl
@@ -1205,7 +1199,7 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 			     << el2b << endl
 			     << el3b << endl
 			     << el4b << endl;
-
+		  */
 
 
 		  mesh[hasbothpoints[0]] = el1b;
@@ -1349,8 +1343,8 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 			  hi3.Sort();
 			  if (faces.Used(hi3))
 			    {
-			      (*testout) << "could improve face conformity, bad1 = " << bad1
-				   << ", bad 2 = " << bad2 << ", nottoobad = " << nottoobad << endl;
+			      // (*testout) << "could improve face conformity, bad1 = " << bad1
+			      // << ", bad 2 = " << bad2 << ", nottoobad = " << nottoobad << endl;
 			      if (nottoobad)
 				confface = l;
 			    }
@@ -1361,8 +1355,10 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 			  if (mesh.BoundaryEdge (suroundpts[l],
 						 suroundpts[k % nsuround]))
 			    {
+			      /*
 			      *testout << "could improve edge conformity, bad1 = " << bad1
 				   << ", bad 2 = " << bad2 << ", nottoobad = " << nottoobad << endl;
+			      */
 			      if (nottoobad)
 				confedge = l;
 			    }
@@ -1390,9 +1386,10 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 		      hel[3] = pi2;
 		      hel.flags.illegal_valid = 0;
 
+		      /*
 		      (*testout) << nsuround << "-swap, new el,top = "
 				 << hel << endl;
-
+		      */
 		      mesh.AddVolumeElement (hel);
 		      for (k1 = 0; k1 < 4; k1++)
 			elementsonnode.Add (hel[k1], mesh.GetNE()-1);
@@ -1402,10 +1399,10 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 		      hel[1] = suroundpts[(k+1) % nsuround];
 		      hel[3] = pi1;
 
-
+		      /*
 		      (*testout) << nsuround << "-swap, new el,bot = "
 				 << hel << endl;
-
+		      */
 
 		      mesh.AddVolumeElement (hel);
 		      for (k1 = 0; k1 < 4; k1++)
@@ -1415,10 +1412,10 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 		  for (k = 0; k < nsuround; k++)
 		    {
 		      Element & rel = mesh[hasbothpoints[k]];
-
+		      /*
 		      (*testout) << nsuround << "-swap, old el = "
 				 << rel << endl;
-
+		      */
 		      rel.Delete();
 		      for (int k1 = 0; k1 < 4; k1++)
 			rel[k1] = 0;
@@ -1451,11 +1448,13 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 
   mesh.Compress ();
 
+  /*
   if (goal == OPT_QUALITY)
     {
       bad1 = CalcTotalBad (mesh.Points(), mesh.VolumeElements());
-      (*testout) << "Total badness = " << bad1 << endl;
+      //      (*testout) << "Total badness = " << bad1 << endl;
     }
+  */
 
   /*
     for (i = 1; i <= GetNE(); i++)
@@ -1486,12 +1485,10 @@ void MeshOptimize3d :: SwapImprove (Mesh & mesh, OPTIMIZEGOAL goal)
 void MeshOptimize3d :: SwapImprove2 (Mesh & mesh, OPTIMIZEGOAL goal)
 {
   int j, k, l;
-  //  INDEX_2 i2;
   ElementIndex ei, eli1, eli2, elnr;
   SurfaceElementIndex sei;
   PointIndex pi1, pi2, pi3, pi4, pi5;
 
-  int mattyp;
   int cnt = 0;
 
   Element el21(TET), el22(TET), el31(TET), el32(TET), el33(TET);
@@ -1530,6 +1527,7 @@ void MeshOptimize3d :: SwapImprove2 (Mesh & mesh, OPTIMIZEGOAL goal)
 
   bad1 = CalcTotalBad (mesh.Points(), mesh.VolumeElements());
   (*testout) << "Total badness = " << bad1 << endl;
+  //  cout << "tot bad = " << bad1 << endl;
 
   // find elements on node
 
@@ -1541,9 +1539,7 @@ void MeshOptimize3d :: SwapImprove2 (Mesh & mesh, OPTIMIZEGOAL goal)
     for (j = 0; j < 3; j++)
       belementsonnode.Add (mesh[sei][j], sei);
 
-  // cout << "main loop" << endl;
-
-  for (eli1 = 0; eli1 < mesh.GetNE(); eli1++)
+  for (eli1 = 0; eli1 < ne; eli1++)
     {
       if (multithread.terminate)
 	break;
@@ -1559,7 +1555,8 @@ void MeshOptimize3d :: SwapImprove2 (Mesh & mesh, OPTIMIZEGOAL goal)
 	  CalcBad (mesh.Points(), mesh[eli1], 0) < 1e3)
 	continue;
 
-
+      // cout << "eli = " << eli1 << endl;
+      //      (*testout) << "swapimp2, eli = " << eli1 << "; el = " << mesh[eli1] << endl;
 
       for (j = 0; j < 4; j++)
 	{
@@ -1569,7 +1566,7 @@ void MeshOptimize3d :: SwapImprove2 (Mesh & mesh, OPTIMIZEGOAL goal)
 	  // if (elem[0] < PointIndex::BASE) continue;
 	  if (elem.IsDeleted()) continue;
 
-	  mattyp = elem.GetIndex();
+	  int mattyp = elem.GetIndex();
 	  
 	  switch (j)
 	    {
@@ -1694,6 +1691,7 @@ void MeshOptimize3d :: SwapImprove2 (Mesh & mesh, OPTIMIZEGOAL goal)
 			   
 		      if (do_swap)
 			{
+			  //			  cout << "do swap, eli1 = " << eli1 << "; eli2 = " << eli2 << endl;
 			  //			  (*mycout) << "2->3 " << flush;
 			  cnt++;
 
@@ -1730,7 +1728,6 @@ void MeshOptimize3d :: SwapImprove2 (Mesh & mesh, OPTIMIZEGOAL goal)
 	}
     }
 
-  //  cout << "loop done" << endl;
 
   PrintMessage (5, cnt, " swaps performed");
 
diff --git a/Netgen/libsrc/meshing/localh.hpp b/Netgen/libsrc/meshing/localh.hpp
index 9d6ea35a50..7531bc7faf 100644
--- a/Netgen/libsrc/meshing/localh.hpp
+++ b/Netgen/libsrc/meshing/localh.hpp
@@ -74,6 +74,8 @@ public:
   ///
   void Delete();
   ///
+  void SetGrading (double agrading) { grading = agrading; }
+  ///
   void SetH (const Point3d & x, double h);
   ///
   double GetH (const Point3d & x) const;
diff --git a/Netgen/libsrc/meshing/meshclass.cpp b/Netgen/libsrc/meshing/meshclass.cpp
index 668c1f3d6e..25c44f337e 100644
--- a/Netgen/libsrc/meshing/meshclass.cpp
+++ b/Netgen/libsrc/meshing/meshclass.cpp
@@ -5,734 +5,982 @@
 namespace netgen
 {
 
-Mesh :: Mesh ()
-  : ident (*this)
+  Mesh :: Mesh ()
+  {
+    boundaryedges = NULL;
+    surfelementht = NULL; 
+    segmentht = NULL;
+
+    lochfunc = NULL;
+    mglevels = 1;
+    elementsearchtree = NULL;
+    elementsearchtreets = NextTimeStamp();
+    majortimestamp = timestamp = NextTimeStamp();
+    hglob = 1e10;
+    numvertices = -1;
+    dimension = 3;
+    topology = new MeshTopology (*this);
+    curvedelems = new CurvedElements (*this);
+    clusters = new AnisotropicClusters (*this);
+    ident = new Identifications (*this);
+
+    hpelements = NULL;
+    coarsemesh = NULL;
 
-{
-  boundaryedges = NULL;
-  surfelementht = NULL; 
-  segmentht = NULL;
-
-  lochfunc = NULL;
-  mglevels = 1;
-  elementsearchtree = NULL;
-  elementsearchtreets = NextTimeStamp();
-  majortimestamp = timestamp = NextTimeStamp();
-  hglob = 1e10;
-  numvertices = -1;
-  dimension = 3;
-  topology = new MeshTopology (*this);
-  curvedelems = new CurvedElements (*this);
-  clusters = new AnisotropicClusters (*this);
-
-  //  volelements.SetName ("Volume Elements");
-  //  surfelements.SetName ("Surface Elements");
-  //  points.SetName ("Mesh Points");
-}
+  }
 
-Mesh :: ~Mesh()
-{
-  // DeleteMesh();
-  delete lochfunc;
-  delete boundaryedges;
-  delete surfelementht;
-  delete segmentht;
-  delete curvedelems;
-  delete clusters;
-  delete topology;
-}
+  Mesh :: ~Mesh()
+  {
+    delete lochfunc;
+    delete boundaryedges;
+    delete surfelementht;
+    delete segmentht;
+    delete curvedelems;
+    delete clusters;
+    delete topology;
+    delete ident;
 
-void Mesh :: DeleteMesh()
-{
-  points.SetSize(0);
-  ptyps.SetSize(0);
-  segments.SetSize(0);
-  surfelements.SetSize(0);
-  volelements.SetSize(0);
-  lockedpoints.SetSize(0);
-  surfacesonnode.SetSize(0);
-
-  delete boundaryedges;
-  boundaryedges = NULL;
-
-  openelements.SetSize(0);
-  facedecoding.SetSize(0);
-
-  ident.Delete();
-
-  delete topology;
-  topology = new MeshTopology (*this);
-  delete curvedelems;
-  curvedelems = new CurvedElements (*this);
-  delete clusters;
-  clusters = new AnisotropicClusters (*this);
-
-  timestamp = NextTimeStamp();
-}
+    delete coarsemesh;
+    delete hpelements;
+    
+    for (int i = 0; i < materials.Size(); i++)
+      delete [] materials[i];
+  }
+
+
+  Mesh & Mesh :: operator= (const Mesh & mesh2)
+  {
+    points = mesh2.points;
+    eltyps = mesh2.eltyps;
+    segments = mesh2.segments;
+    surfelements = mesh2.surfelements;
+    volelements = mesh2.volelements;
+    lockedpoints = mesh2.lockedpoints;
+    facedecoding = mesh2.facedecoding;
+    dimension = mesh2.dimension;
+    return *this;
+  }
+
+  void Mesh :: DeleteMesh()
+  {
+    points.SetSize(0);
+    // ptyps.SetSize(0);
+    segments.SetSize(0);
+    surfelements.SetSize(0);
+    volelements.SetSize(0);
+    lockedpoints.SetSize(0);
+    surfacesonnode.SetSize(0);
 
+    delete boundaryedges;
+    boundaryedges = NULL;
 
+    openelements.SetSize(0);
+    facedecoding.SetSize(0);
 
-PointIndex Mesh :: AddPoint (const Point3d & p, int layer)
-{ 
-  NgLock lock(mutex);
-  lock.Lock();
+    // ident -> Delete();
+    delete ident;
+    ident = new Identifications (*this);
+    delete topology;
+    topology = new MeshTopology (*this);
+    delete curvedelems;
+    curvedelems = new CurvedElements (*this);
+    delete clusters;
+    clusters = new AnisotropicClusters (*this);
 
-  timestamp = NextTimeStamp();
+    timestamp = NextTimeStamp();
+  }
 
-  if (ptyps.Size() == points.Size())
-    ptyps.Append (INNERPOINT);
 
-  PointIndex pi = points.Size() + PointIndex::BASE;
-  points.Append ( MeshPoint (p, layer) ); 
 
-  lock.UnLock();
+  PointIndex Mesh :: AddPoint (const Point3d & p, int layer)
+  { 
+    NgLock lock(mutex);
+    lock.Lock();
 
-  return pi;
-}
+    timestamp = NextTimeStamp();
+
+    PointIndex pi = points.Size() + PointIndex::BASE;
+    points.Append ( MeshPoint (p, layer, INNERPOINT) ); 
 
+    lock.UnLock();
 
-SegmentIndex Mesh :: AddSegment (const Segment & s)
-{ 
-  NgLock lock(mutex);	
-  lock.Lock();
-  timestamp = NextTimeStamp();
+    return pi;
+  }
 
-  int maxn = max2 (s.p1, s.p2);
-  maxn += 1-PointIndex::BASE;
 
-  if (maxn > ptyps.Size())
-    {
+  SegmentIndex Mesh :: AddSegment (const Segment & s)
+  { 
+    NgLock lock(mutex);	
+    lock.Lock();
+    timestamp = NextTimeStamp();
+
+    int maxn = max2 (s.p1, s.p2);
+    maxn += 1-PointIndex::BASE;
+
+    /*
+      if (maxn > ptyps.Size())
+      {
       int maxo = ptyps.Size();
       ptyps.SetSize (maxn);
       for (int i = maxo; i < maxn; i++)
-	ptyps[i] = INNERPOINT;
-    }
+      ptyps[i] = INNERPOINT;
+      }
 
-  if (ptyps[s.p1] > EDGEPOINT) ptyps[s.p1] = EDGEPOINT;
-  if (ptyps[s.p2] > EDGEPOINT) ptyps[s.p2] = EDGEPOINT;
-  
-  SegmentIndex si = segments.Size();
-  segments.Append (s); 
+      if (ptyps[s.p1] > EDGEPOINT) ptyps[s.p1] = EDGEPOINT;
+      if (ptyps[s.p2] > EDGEPOINT) ptyps[s.p2] = EDGEPOINT;
+    */
+
+    if (maxn <= points.Size())
+      {
+	if (points[s.p1].Type() > EDGEPOINT)
+	  points[s.p1].SetType (EDGEPOINT);
+	if (points[s.p2].Type() > EDGEPOINT)
+	  points[s.p2].SetType (EDGEPOINT);
+      }
+    /*
+      else
+      {
+      cerr << "edge points nrs > points.Size" << endl;
+      }
+    */
+
+    SegmentIndex si = segments.Size();
+    segments.Append (s); 
   
-  lock.UnLock();
-  return si;
-}
+    lock.UnLock();
+    return si;
+  }
 
-SurfaceElementIndex Mesh :: AddSurfaceElement (const Element2d & el)
-{     
-  NgLock lock(mutex);
-  lock.Lock();
-  timestamp = NextTimeStamp();
+  SurfaceElementIndex Mesh :: AddSurfaceElement (const Element2d & el)
+  {     
+    NgLock lock(mutex);
+    lock.Lock();
+    timestamp = NextTimeStamp();
 
-  int i;
-  int maxn = el[0];
-  for (i = 1; i < el.GetNP(); i++)
-    if (el[i] > maxn) maxn = el[i];
+    int maxn = el[0];
+    for (int i = 1; i < el.GetNP(); i++)
+      if (el[i] > maxn) maxn = el[i];
 
-  maxn += 1-PointIndex::BASE;
+    maxn += 1-PointIndex::BASE;
 
-  if (maxn > ptyps.Size())
-    {
+    /*
+      if (maxn > ptyps.Size())
+      {
       int maxo = ptyps.Size();
       ptyps.SetSize (maxn);
       for (i = maxo+PointIndex::BASE; 
-	   i < maxn+PointIndex::BASE; i++)
-	ptyps[i] = INNERPOINT;
-    }
-
-  for (i = 0; i < el.GetNP(); i++)
-    if (ptyps[el[i]] > SURFACEPOINT)
-      ptyps[el[i]] = SURFACEPOINT;
-
-  SurfaceElementIndex si = surfelements.Size();
-  surfelements.Append (el); 
+      i < maxn+PointIndex::BASE; i++)
+      ptyps[i] = INNERPOINT;
+      
+      }
+    */
+    if (maxn <= points.Size())
+      {
+	for (int i = 0; i < el.GetNP(); i++)
+	  if (points[el[i]].Type() > SURFACEPOINT)
+	    points[el[i]].SetType(SURFACEPOINT);
+      }
+    /*
+      else
+      {
+      cerr << "surf points nrs > points.Size" << endl;      
+      }
+    */
 
-  lock.UnLock();
-  return si;
-}
+    SurfaceElementIndex si = surfelements.Size();
+    surfelements.Append (el); 
 
+    lock.UnLock();
+    return si;
+  }
 
-ElementIndex Mesh :: AddVolumeElement (const Element & el)
-{ 
-  NgLock lock(mutex);
-  lock.Lock();
 
-  int i;
+  ElementIndex Mesh :: AddVolumeElement (const Element & el)
+  { 
+    NgLock lock(mutex);
+    lock.Lock();
 
-  int maxn = el[0];
-  for (i = 1; i < el.GetNP(); i++)
-    if (el[i] > maxn) maxn = el[i];
+    int maxn = el[0];
+    for (int i = 1; i < el.GetNP(); i++)
+      if (el[i] > maxn) maxn = el[i];
 
-  maxn += 1-PointIndex::BASE;
+    maxn += 1-PointIndex::BASE;
 
-  if (maxn > ptyps.Size())
-    {
+    /*
+      if (maxn > ptyps.Size())
+      {
       int maxo = ptyps.Size();
       ptyps.SetSize (maxn);
       for (i = maxo+PointIndex::BASE; 
-	   i < maxn+PointIndex::BASE; i++)
-	ptyps[i] = INNERPOINT;
-    }
-
+      i < maxn+PointIndex::BASE; i++)
+      ptyps[i] = INNERPOINT;
+      }
+    */
+    /*
+      if (maxn > points.Size())
+      {
+      cerr << "add vol element before point" << endl;
+      }
+    */
 
-  int ve = volelements.Size();
+    int ve = volelements.Size();
 
-  volelements.Append (el); 
-  volelements.Last().flags.illegal_valid = 0;
+    volelements.Append (el); 
+    volelements.Last().flags.illegal_valid = 0;
 
-  while (volelements.Size() > eltyps.Size())
-    eltyps.Append (FREEELEMENT);
+    while (volelements.Size() > eltyps.Size())
+      eltyps.Append (FREEELEMENT);
   
-  timestamp = NextTimeStamp();
-
-  lock.UnLock();
-  return ve;
-}
+    timestamp = NextTimeStamp();
 
+    lock.UnLock();
+    return ve;
+  }
 
 
 
 
 
-void Mesh :: Save (const char * filename) const
-{
-  int i, j;
 
-  double scale = 1;  // globflags.GetNumFlag ("scale", 1);
-  int inverttets = 0;  // globflags.GetDefineFlag ("inverttets");
-  int invertsurf = 0;  // globflags.GetDefineFlag ("invertsurfacemesh");
+  void Mesh :: Save (const string & filename) const
+  {
+    int i, j;
 
-  ofstream outfile(filename);
+    double scale = 1;  // globflags.GetNumFlag ("scale", 1);
+    int inverttets = 0;  // globflags.GetDefineFlag ("inverttets");
+    int invertsurf = 0;  // globflags.GetDefineFlag ("invertsurfacemesh");
 
+    ofstream outfile(filename.c_str());
 
-  outfile << "mesh3d" << "\n";
 
-  outfile << "dimension\n" << GetDimension() << "\n";
+    outfile << "mesh3d" << "\n";
 
-  outfile << "\n";
-  outfile << "# surfnr    bcnr   domin  domout      np      p1      p2      p3"
-	  << "\n";
+    outfile << "dimension\n" << GetDimension() << "\n";
 
-  outfile << "surfaceelementsgi" << "\n";
-  //  outfile << "surfaceelements" << "\n";
-  outfile << GetNSE() << "\n";
+    outfile << "\n";
+    outfile << "# surfnr    bcnr   domin  domout      np      p1      p2      p3"
+	    << "\n";
 
-  SurfaceElementIndex sei;
-  for (sei = 0; sei < GetNSE(); sei++)
-    {
-      if ((*this)[sei].GetIndex())
-	{
-	  outfile.width(8);
-	  outfile << GetFaceDescriptor((*this)[sei].GetIndex ()).SurfNr()+1;
-	  outfile.width(8);
-	  outfile << GetFaceDescriptor((*this)[sei].GetIndex ()).BCProperty();
-	  outfile.width(8);	  
-	  outfile << GetFaceDescriptor((*this)[sei].GetIndex ()).DomainIn();
-	  outfile.width(8);	  
-	  outfile << GetFaceDescriptor((*this)[sei].GetIndex ()).DomainOut();
-	}
-      else
-	outfile << "       0       0       0";
+    outfile << "surfaceelementsgi" << "\n";
+    //  outfile << "surfaceelements" << "\n";
+    outfile << GetNSE() << "\n";
 
+    SurfaceElementIndex sei;
+    for (sei = 0; sei < GetNSE(); sei++)
+      {
+	if ((*this)[sei].GetIndex())
+	  {
+	    outfile.width(8);
+	    outfile << GetFaceDescriptor((*this)[sei].GetIndex ()).SurfNr()+1;
+	    outfile.width(8);
+	    outfile << GetFaceDescriptor((*this)[sei].GetIndex ()).BCProperty();
+	    outfile.width(8);	  
+	    outfile << GetFaceDescriptor((*this)[sei].GetIndex ()).DomainIn();
+	    outfile.width(8);	  
+	    outfile << GetFaceDescriptor((*this)[sei].GetIndex ()).DomainOut();
+	  }
+	else
+	  outfile << "       0       0       0";
 
-      Element2d sel = (*this)[sei];
-      if (invertsurf)
-	sel.Invert();
 
-      outfile.width(8);
-      outfile << sel.GetNP();
+	Element2d sel = (*this)[sei];
+	if (invertsurf)
+	  sel.Invert();
 
-      for (j = 0; j < sel.GetNP(); j++)
-	{
-	  outfile.width(8);	  
-	  outfile << sel[j];
-	}
+	outfile.width(8);
+	outfile << sel.GetNP();
 
-      for (j = 1; j <= sel.GetNP(); j++)
-	{
-	  outfile.width(7);	  
-	  outfile << " " << sel.GeomInfoPi(j).trignum;
-	}
-      outfile << endl;
-    }
+	for (j = 0; j < sel.GetNP(); j++)
+	  {
+	    outfile.width(8);	  
+	    outfile << sel[j];
+	  }
 
-  outfile << "\n" << "\n";
-  outfile << "#  matnr      np      p1      p2      p3      p4" << "\n";
-  outfile << "volumeelements" << "\n";
-  outfile << GetNE() << "\n";
+	for (j = 1; j <= sel.GetNP(); j++)
+	  {
+	    outfile.width(7);	  
+	    outfile << " " << sel.GeomInfoPi(j).trignum;
+	  }
+	outfile << endl;
+      }
 
-  for (ElementIndex ei = 0; ei < GetNE(); ei++)
-    {
-      outfile.width(8);
-      outfile << (*this)[ei].GetIndex();
-      outfile.width(8);
-      outfile << (*this)[ei].GetNP();
+    outfile << "\n" << "\n";
+    outfile << "#  matnr      np      p1      p2      p3      p4" << "\n";
+    outfile << "volumeelements" << "\n";
+    outfile << GetNE() << "\n";
 
-      Element el = (*this)[ei];
-      if (inverttets)
-	el.Invert();
+    for (ElementIndex ei = 0; ei < GetNE(); ei++)
+      {
+	outfile.width(8);
+	outfile << (*this)[ei].GetIndex();
+	outfile.width(8);
+	outfile << (*this)[ei].GetNP();
 
-      for (j = 0; j < el.GetNP(); j++)
-	{
-	  outfile.width(8);
-	  outfile << el[j];
-	}
-      outfile << "\n";
-    }
+	Element el = (*this)[ei];
+	if (inverttets)
+	  el.Invert();
 
+	for (j = 0; j < el.GetNP(); j++)
+	  {
+	    outfile.width(8);
+	    outfile << el[j];
+	  }
+	outfile << "\n";
+      }
 
-  outfile << "\n" << "\n";
-  outfile << "   surf1   surf2      p1      p2" << "\n";
-  outfile << "edgesegmentsgi2" << "\n";
-  outfile << GetNSeg() << "\n";
 
-  for (i = 1; i <= GetNSeg(); i++)
-    {
-      const Segment & seg = LineSegment (i);
-      outfile.width(8);
-      outfile << seg.si;
-      outfile.width(8);
-      outfile << 0;
-      outfile.width(8);
-      outfile << seg.p1;
-      outfile.width(8);
-      outfile << seg.p2;
-      outfile << " ";
-      outfile.width(8);
-      outfile << seg.geominfo[0].trignum;
-      outfile << " ";
-      outfile.width(8);
-      outfile << seg.geominfo[1].trignum << endl;
-
-      outfile << " ";
-      outfile.width(8);
-      outfile << seg.surfnr1+1;
-      outfile << " ";
-      outfile.width(8);
-      outfile << seg.surfnr2+1;
-      outfile << " ";
-      outfile.width(8);
-      outfile << seg.edgenr;
-      //      outfile << seg.epgeominfo[0].edgenr;
-      /*
-      outfile.width(8);
-      outfile << seg.epgeominfo[0].lefttrig;
-      outfile.width(8);
-      outfile << seg.epgeominfo[0].righttrig;
-      */
-      outfile << " ";
-      outfile.width(12);
-      outfile << seg.epgeominfo[0].dist;
-      outfile << " ";
-      outfile.width(8);
-      outfile << seg.epgeominfo[1].edgenr;
-      /*
-      outfile.width(8);
-      outfile << seg.epgeominfo[1].lefttrig;
-      outfile.width(8);
-      outfile << seg.epgeominfo[1].righttrig;
-      */
-      outfile << " ";
-      outfile.width(12);
-      outfile << seg.epgeominfo[1].dist;
-
-      outfile << "\n";
-    }
+    outfile << "\n" << "\n";
+    outfile << "   surf1   surf2      p1      p2" << "\n";
+    outfile << "edgesegmentsgi2" << "\n";
+    outfile << GetNSeg() << "\n";
 
+    for (i = 1; i <= GetNSeg(); i++)
+      {
+	const Segment & seg = LineSegment (i);
+	outfile.width(8);
+	outfile << seg.si;
+	outfile.width(8);
+	outfile << 0;
+	outfile.width(8);
+	outfile << seg.p1;
+	outfile.width(8);
+	outfile << seg.p2;
+	outfile << " ";
+	outfile.width(8);
+	outfile << seg.geominfo[0].trignum;
+	outfile << " ";
+	outfile.width(8);
+	outfile << seg.geominfo[1].trignum << endl;
+
+	outfile << " ";
+	outfile.width(8);
+	outfile << seg.surfnr1+1;
+	outfile << " ";
+	outfile.width(8);
+	outfile << seg.surfnr2+1;
+	outfile << " ";
+	outfile.width(8);
+	outfile << seg.edgenr;
+	//      outfile << seg.epgeominfo[0].edgenr;
+	/*
+	  outfile.width(8);
+	  outfile << seg.epgeominfo[0].lefttrig;
+	  outfile.width(8);
+	  outfile << seg.epgeominfo[0].righttrig;
+	*/
+	outfile << " ";
+	outfile.width(12);
+	outfile << seg.epgeominfo[0].dist;
+	outfile << " ";
+	outfile.width(8);
+	outfile << seg.epgeominfo[1].edgenr;
+	/*
+	  outfile.width(8);
+	  outfile << seg.epgeominfo[1].lefttrig;
+	  outfile.width(8);
+	  outfile << seg.epgeominfo[1].righttrig;
+	*/
+	outfile << " ";
+	outfile.width(12);
+	outfile << seg.epgeominfo[1].dist;
 
+	outfile << "\n";
+      }
 
 
-  outfile << "\n" << "\n";
-  outfile << "#          X             Y             Z" << "\n";
-  outfile << "points" << "\n";
-  outfile << GetNP() << "\n";
-  outfile.precision(16);
-  outfile.setf (ios::fixed, ios::floatfield);
-  outfile.setf (ios::showpoint);
+    outfile << "\n" << "\n";
+    outfile << "#          X             Y             Z" << "\n";
+    outfile << "points" << "\n";
+    outfile << GetNP() << "\n";
+    outfile.precision(16);
+    outfile.setf (ios::fixed, ios::floatfield);
+    outfile.setf (ios::showpoint);
 
-  PointIndex pi;
-  for (pi = 1; pi <= GetNP(); pi++)
-    {
-      outfile.width(22);
-      outfile << (*this)[pi].X()/scale << "  ";
-      outfile.width(22);
-      outfile << (*this)[pi].Y()/scale << "  ";
-      outfile.width(22);
-      outfile << (*this)[pi].Z()/scale << "\n";
-    }      
-
-  if (ident.GetMaxNr() > 0)
-    {
-      outfile << "identifications\n";
-      ARRAY<INDEX_2> identpairs;
-      int cnt = 0;
-      for (i = 1; i <= ident.GetMaxNr(); i++)
-	{
-	  ident.GetPairs (i, identpairs);
-	  cnt += identpairs.Size();
-	}
-      outfile << cnt << "\n";
-      for (i = 1; i <= ident.GetMaxNr(); i++)
-	{
-	  ident.GetPairs (i, identpairs);
-	  for (j = 1; j <= identpairs.Size(); j++)
-	    {
-	      outfile.width (8);
-	      outfile << identpairs.Get(j).I1();
-	      outfile.width (8);
-	      outfile << identpairs.Get(j).I2();
-	      outfile.width (8);
-	      outfile << i << "\n";
-	    }
-	}
-    }
+    PointIndex pi;
+    for (pi = PointIndex::BASE; 
+	 pi < GetNP()+PointIndex::BASE; pi++)
+      {
+	outfile.width(22);
+	outfile << (*this)[pi].X()/scale << "  ";
+	outfile.width(22);
+	outfile << (*this)[pi].Y()/scale << "  ";
+	outfile.width(22);
+	outfile << (*this)[pi].Z()/scale << "\n";
+      }      
+
+    if (ident -> GetMaxNr() > 0)
+      {
+	outfile << "identifications\n";
+	ARRAY<INDEX_2> identpairs;
+	int cnt = 0;
+	for (i = 1; i <= ident -> GetMaxNr(); i++)
+	  {
+	    ident -> GetPairs (i, identpairs);
+	    cnt += identpairs.Size();
+	  }
+	outfile << cnt << "\n";
+	for (i = 1; i <= ident -> GetMaxNr(); i++)
+	  {
+	    ident -> GetPairs (i, identpairs);
+	    for (j = 1; j <= identpairs.Size(); j++)
+	      {
+		outfile.width (8);
+		outfile << identpairs.Get(j).I1();
+		outfile.width (8);
+		outfile << identpairs.Get(j).I2();
+		outfile.width (8);
+		outfile << i << "\n";
+	      }
+	  }
+      }
 
-  int cntmat = 0;
-  for (i = 1; i <= materials.Size(); i++)
-    if (materials.Get(i) && strlen (materials.Get(i)))
-      cntmat++;
+    int cntmat = 0;
+    for (i = 1; i <= materials.Size(); i++)
+      if (materials.Get(i) && strlen (materials.Get(i)))
+	cntmat++;
 
-  if (cntmat)
-    {
-      outfile << "materials" << endl;
-      outfile << cntmat << endl;
-      for (i = 1; i <= materials.Size(); i++)
-	if (materials.Get(i) && strlen (materials.Get(i)))
-	  outfile << i << " " << materials.Get(i) << endl;
-    }
+    if (cntmat)
+      {
+	outfile << "materials" << endl;
+	outfile << cntmat << endl;
+	for (i = 1; i <= materials.Size(); i++)
+	  if (materials.Get(i) && strlen (materials.Get(i)))
+	    outfile << i << " " << materials.Get(i) << endl;
+      }
 
-  //  delete houtfile;
-}
+    
+    int cnt_sing = 0;
+    for (PointIndex pi = PointIndex::BASE; pi < GetNP()+PointIndex::BASE; pi++)
+      if ((*this)[pi].IsSingular()) cnt_sing++;
+    
+    if (cnt_sing)
+      {
+	outfile << "singular_points" << endl << cnt_sing << endl;
+	for (PointIndex pi = PointIndex::BASE; pi < GetNP()+PointIndex::BASE; pi++)
+	  if ((*this)[pi].IsSingular()) 
+	    outfile << int(pi) << endl;
+      }
 
+    cnt_sing = 0;
+    for (SegmentIndex si = 0; si < GetNSeg(); si++)
+      if ( segments[si].singedge_left ) cnt_sing++;
+    if (cnt_sing)
+      {
+	outfile << "singular_edge_left" << endl << cnt_sing << endl;
+	for (SegmentIndex si = 0; si < GetNSeg(); si++)
+	  if ( segments[si].singedge_left )
+	    outfile << int(si) << endl;
+      }
 
-void Mesh :: Load (const char * filename)
-{
-  char str[100];
-  int i, n;
+    cnt_sing = 0;
+    for (SegmentIndex si = 0; si < GetNSeg(); si++)
+      if ( segments[si].singedge_right ) cnt_sing++;
+    if (cnt_sing)
+      {
+	outfile << "singular_edge_right" << endl << cnt_sing << endl;
+	for (SegmentIndex si = 0; si < GetNSeg(); si++)
+	  if ( segments[si].singedge_right )
+	    outfile << int(si) << endl;
+      }
 
-  double scale = 1;  // globflags.GetNumFlag ("scale", 1);
-  int inverttets = 0;  // globflags.GetDefineFlag ("inverttets");
-  int invertsurf = 0;  // globflags.GetDefineFlag ("invertsurfacemesh");
 
+    cnt_sing = 0;
+    for (SurfaceElementIndex sei = 0; sei < GetNSE(); sei++)
+      if ( GetFaceDescriptor ((*this)[sei].GetIndex()).domin_singular) cnt_sing++;
+    if (cnt_sing)
+      {
+	outfile << "singular_face_inside" << endl << cnt_sing << endl;
+	for (SurfaceElementIndex sei = 0; sei < GetNSE(); sei++)
+	  if ( GetFaceDescriptor ((*this)[sei].GetIndex()).domin_singular) 
+	    outfile << int(sei) << endl;
+      }
 
-  ifstream infile(filename);
-  if (!infile.good())
-    {
-      throw NgException ("mesh file not found");
-      //      cerr << "mesh file not found !!" << endl;
-      //      return;
-    }
+    cnt_sing = 0;
+    for (SurfaceElementIndex sei = 0; sei < GetNSE(); sei++)
+      if ( GetFaceDescriptor ((*this)[sei].GetIndex()).domout_singular) cnt_sing++;
+    if (cnt_sing)
+      {
+	outfile << "singular_face_outside" << endl << cnt_sing << endl;
+	for (SurfaceElementIndex sei = 0; sei < GetNSE(); sei++)
+	  if ( GetFaceDescriptor ((*this)[sei].GetIndex()).domout_singular) 
+	    outfile << int(sei) << endl;
+      }
 
-  facedecoding.SetSize(0);
 
-  while (infile.good())
-    {
-      infile >> str;
+  }
 
-      if (strcmp (str, "dimension") == 0)
-	{
-	  infile >> dimension;
-	}
 
-      if (strcmp (str, "surfaceelements") == 0)
-	{
-	  infile >> n;
-	  PrintMessage (3, n, " surface elements");
-	  for (i = 1; i <= n; i++)
-	    {
-	      int j;
-	      int surfnr, bcp, domin, domout, nep, faceind = 0;
+  void Mesh :: Load (const string & filename)
+  {
+    char str[100];
+    int i, n;
 
-	      infile >> surfnr >> bcp >> domin >> domout;
-	      surfnr--;
+    double scale = 1;  // globflags.GetNumFlag ("scale", 1);
+    int inverttets = 0;  // globflags.GetDefineFlag ("inverttets");
+    int invertsurf = 0;  // globflags.GetDefineFlag ("invertsurfacemesh");
 
-	      for (j = 1; j <= facedecoding.Size(); j++)
-		if (GetFaceDescriptor(j).SurfNr() == surfnr &&
-		    GetFaceDescriptor(j).BCProperty() == bcp &&
-		    GetFaceDescriptor(j).DomainIn() == domin &&
-		    GetFaceDescriptor(j).DomainOut() == domout)
-		  faceind = j;
+    ifstream infile(filename.c_str());
+    if (!infile.good())
+      throw NgException ("mesh file not found");
 
-	      if (!faceind)
-		{
-		  faceind = AddFaceDescriptor (FaceDescriptor(surfnr, domin, domout, 0));
-		  GetFaceDescriptor(faceind).SetBCProperty (bcp);
-		}
+    facedecoding.SetSize(0);
 
-	      infile >> nep;
-	      if (!nep) nep = 3;
+    while (infile.good())
+      {
+	infile >> str;
 
-	      Element2d tri(nep);
-	      tri.SetIndex(faceind);
+	if (strcmp (str, "dimension") == 0)
+	  {
+	    infile >> dimension;
+	  }
 
-	      for (j = 1; j <= nep; j++)
-		infile >> tri.PNum(j);
+	if (strcmp (str, "surfaceelements") == 0)
+	  {
+	    infile >> n;
+	    PrintMessage (3, n, " surface elements");
+	    for (i = 1; i <= n; i++)
+	      {
+		int j;
+		int surfnr, bcp, domin, domout, nep, faceind = 0;
 
-	      if (invertsurf)
-		tri.Invert();
+		infile >> surfnr >> bcp >> domin >> domout;
+		surfnr--;
 
-	      AddSurfaceElement (tri);
-	    }
-	}
+		for (j = 1; j <= facedecoding.Size(); j++)
+		  if (GetFaceDescriptor(j).SurfNr() == surfnr &&
+		      GetFaceDescriptor(j).BCProperty() == bcp &&
+		      GetFaceDescriptor(j).DomainIn() == domin &&
+		      GetFaceDescriptor(j).DomainOut() == domout)
+		    faceind = j;
 
-      if (strcmp (str, "surfaceelementsgi") == 0)
-	{
-	  infile >> n;
-	  PrintMessage (3, n, " surface elements");
-	  for (i = 1; i <= n; i++)
-	    {
-	      int j;
-	      int surfnr, bcp, domin, domout, nep, faceind = 0;
-	      infile >> surfnr >> bcp >> domin >> domout;
-	      surfnr--;
-
-	      for (j = 1; j <= facedecoding.Size(); j++)
-		if (GetFaceDescriptor(j).SurfNr() == surfnr &&
-		    GetFaceDescriptor(j).BCProperty() == bcp &&
-		    GetFaceDescriptor(j).DomainIn() == domin &&
-		    GetFaceDescriptor(j).DomainOut() == domout)
-		  faceind = j;
-
-	      if (!faceind)
-		{
-		  faceind = AddFaceDescriptor (FaceDescriptor(surfnr, domin, domout, 0));
-		  GetFaceDescriptor(faceind).SetBCProperty (bcp);
-		}
+		if (!faceind)
+		  {
+		    faceind = AddFaceDescriptor (FaceDescriptor(surfnr, domin, domout, 0));
+		    GetFaceDescriptor(faceind).SetBCProperty (bcp);
+		  }
 
-	      infile >> nep;
-	      if (!nep) nep = 3;
+		infile >> nep;
+		if (!nep) nep = 3;
 
-	      Element2d tri(nep);
-	      tri.SetIndex(faceind);
+		Element2d tri(nep);
+		tri.SetIndex(faceind);
 
-	      for (j = 1; j <= nep; j++)
-		infile >> tri.PNum(j);
+		for (j = 1; j <= nep; j++)
+		  infile >> tri.PNum(j);
 
-	      for (j = 1; j <= nep; j++)
-		infile >> tri.GeomInfoPi(j).trignum;
+		if (invertsurf)
+		  tri.Invert();
 
-	      if (invertsurf)
-		tri.Invert();
+		AddSurfaceElement (tri);
+	      }
+	  }
+      
+	if (strcmp (str, "surfaceelementsgi") == 0)
+	  {
+	    infile >> n;
+	    PrintMessage (3, n, " surface elements");
+	    for (i = 1; i <= n; i++)
+	      {
+		int j;
+		int surfnr, bcp, domin, domout, nep, faceind = 0;
+		infile >> surfnr >> bcp >> domin >> domout;
+		surfnr--;
+
+		for (j = 1; j <= facedecoding.Size(); j++)
+		  if (GetFaceDescriptor(j).SurfNr() == surfnr &&
+		      GetFaceDescriptor(j).BCProperty() == bcp &&
+		      GetFaceDescriptor(j).DomainIn() == domin &&
+		      GetFaceDescriptor(j).DomainOut() == domout)
+		    faceind = j;
+
+		if (!faceind)
+		  {
+		    faceind = AddFaceDescriptor (FaceDescriptor(surfnr, domin, domout, 0));
+		    GetFaceDescriptor(faceind).SetBCProperty (bcp);
+		  }
 
-	      AddSurfaceElement (tri);
-	    }
-	}
+		infile >> nep;
+		if (!nep) nep = 3;
 
+		Element2d tri(nep);
+		tri.SetIndex(faceind);
 
+		for (j = 1; j <= nep; j++)
+		  infile >> tri.PNum(j);
 
-      if (strcmp (str, "volumeelements") == 0)
-	{
-	  infile >> n;
-	  PrintMessage (3, n, " volume elements");
-	  for (i = 1; i <= n; i++)
-	    {
-	      Element el;
-	      int hi, nep;
-	      infile >> hi;
-	      if (hi == 0) hi = 1;
-	      el.SetIndex(hi);
-	      infile >> nep;
-	      el.SetNP(nep);
-	      
-	      for (int j = 0; j < nep; j++)
-		infile >> (int&)(el[j]);
-	      
-	      if (inverttets)
-		el.Invert();
+		for (j = 1; j <= nep; j++)
+		  infile >> tri.GeomInfoPi(j).trignum;
 
-	      AddVolumeElement (el);
-	      
-	    }
-	}
+		if (invertsurf)
+		  tri.Invert();
 
-      if (strcmp (str, "edgesegments") == 0)
-	{
-	  infile >> n;
-	  for (i = 1; i <= n; i++)
-	    {
-	      Segment seg;
-	      int hi;
-	      infile >> seg.si >> hi >> seg.p1 >> seg.p2;
-	      AddSegment (seg);
-	    }
-	}
-      
-      if (strcmp (str, "edgesegmentsgi") == 0)
-	{
-	  infile >> n;
-	  for (i = 1; i <= n; i++)
-	    {
-	      Segment seg;
-	      int hi;
-	      infile >> seg.si >> hi >> seg.p1 >> seg.p2
-		     >> seg.geominfo[0].trignum
-		     >> seg.geominfo[1].trignum;
-	      AddSegment (seg);
-	    }
-	}
-      if (strcmp (str, "edgesegmentsgi2") == 0)
-	{
-	  infile >> n;
-	  PrintMessage (3, n, " curve elements");
+		AddSurfaceElement (tri);
+	      }
+	  }
 
-	  for (i = 1; i <= n; i++)
-	    {
-	      Segment seg;
-	      int hi;
-	      infile >> seg.si >> hi >> seg.p1 >> seg.p2
-		     >> seg.geominfo[0].trignum
-		     >> seg.geominfo[1].trignum
-		     >> seg.surfnr1 >> seg.surfnr2
-		     >> seg.epgeominfo[0].edgenr
-		//		     >> seg.epgeominfo[0].lefttrig
-		//		     >> seg.epgeominfo[0].righttrig
-		     >> seg.epgeominfo[0].dist
-		     >> seg.epgeominfo[1].edgenr
-		//		     >> seg.epgeominfo[1].lefttrig
-		//		     >> seg.epgeominfo[1].righttrig
-		     >> seg.epgeominfo[1].dist;
-	      seg.edgenr = seg.epgeominfo[0].edgenr;
-
-	      seg.surfnr1--;
-	      seg.surfnr2--;
+	if (strcmp (str, "volumeelements") == 0)
+	  {
+	    infile >> n;
+	    PrintMessage (3, n, " volume elements");
+	    for (i = 1; i <= n; i++)
+	      {
+		Element el;
+		int hi, nep;
+		infile >> hi;
+		if (hi == 0) hi = 1;
+		el.SetIndex(hi);
+		infile >> nep;
+		el.SetNP(nep);
 	      
-	      AddSegment (seg);
-	    }
-	}
-      
-      if (strcmp (str, "points") == 0)
-	{
-	  infile >> n;
-	  PrintMessage (3, n, " points");
-	  for (i = 1; i <= n; i++)
-	    {
-	      Point3d p;
-	      infile >> p.X() >> p.Y() >> p.Z();
-	      p.X() *= scale;
-	      p.Y() *= scale;
-	      p.Z() *= scale;
-	      AddPoint (p);
-	    }
-	}
-
-      if (strcmp (str, "identifications") == 0)
-	{
-	  infile >> n;
-	  for (i = 1; i <= n; i++)
-	    {
-	      PointIndex pi1, pi2;
-	      int ind;
-	      infile >> pi1 >> pi2 >> ind;
-	      ident.Add (pi1, pi2, ind);
-	    }
-	}
-      if (strcmp (str, "materials") == 0)
-	{
-	  infile >> n;
-	  for (i = 1; i <= n; i++)
-	    {
-	      int nr;
-	      string mat;
-	      infile >> nr >> mat;
-	      SetMaterial (nr, mat.c_str());
-	    }
-	}
+		for (int j = 0; j < nep; j++)
+		  infile >> (int&)(el[j]);
+	      
+		if (inverttets)
+		  el.Invert();
 
-      strcpy (str, "");
-    }
-  
-  CalcSurfacesOfNode ();
-  //  BuildConnectedNodes ();
-  topology -> Update();
-  clusters -> Update();
-  
-  SetNextMajorTimeStamp();
-  //  PrintMemInfo (cout);
-}
-  
-   
+		AddVolumeElement (el);
+	      }
+	  }
+    
 
-bool Mesh :: TestOk () const
-{
-  for (ElementIndex ei = 0; ei < volelements.Size(); ei++)
-    {
-      for (int j = 0; j < 4; j++)
-	if ( (*this)[ei][j] <= PointIndex::BASE-1)
+	if (strcmp (str, "edgesegments") == 0)
 	  {
-	    (*testout) << "El " << ei << " has 0 nodes: ";
-	    for (int k = 0; k < 4; k++)
-	      (*testout) << (*this)[ei][k];
-	    break;
+	    infile >> n;
+	    for (i = 1; i <= n; i++)
+	      {
+		Segment seg;
+		int hi;
+		infile >> seg.si >> hi >> seg.p1 >> seg.p2;
+		AddSegment (seg);
+	      }
 	  }
-    }
-  CheckMesh3D (*this);
-  return 1;
-}
+      
 
-void Mesh :: CalcSurfacesOfNode ()
-{
-  int i, j, k;
-  SurfaceElementIndex sei;
 
-  surfacesonnode.SetSize (GetNP());
-  if (boundaryedges)
-    delete boundaryedges;
-  if (surfelementht)
-    delete surfelementht;
-  if (segmentht)
-    delete segmentht;
+	if (strcmp (str, "edgesegmentsgi") == 0)
+	  {
+	    infile >> n;
+	    for (i = 1; i <= n; i++)
+	      {
+		Segment seg;
+		int hi;
+		infile >> seg.si >> hi >> seg.p1 >> seg.p2
+		       >> seg.geominfo[0].trignum
+		       >> seg.geominfo[1].trignum;
+		AddSegment (seg);
+	      }
+	  }
+	if (strcmp (str, "edgesegmentsgi2") == 0)
+	  {
+	    infile >> n;
+	    PrintMessage (3, n, " curve elements");
 
-  boundaryedges = new INDEX_2_CLOSED_HASHTABLE<int>
-    (3 * (GetNSE() + GetNOpenElements()) + GetNSeg() + 1);
-  surfelementht = new INDEX_3_HASHTABLE<int> (GetNSE()/4 + 1);
-  segmentht = new INDEX_2_HASHTABLE<int> (GetNSeg() + 1);
+	    for (i = 1; i <= n; i++)
+	      {
+		Segment seg;
+		int hi;
+		infile >> seg.si >> hi >> seg.p1 >> seg.p2
+		       >> seg.geominfo[0].trignum
+		       >> seg.geominfo[1].trignum
+		       >> seg.surfnr1 >> seg.surfnr2
+		       >> seg.edgenr
+		  // >> seg.epgeominfo[0].edgenr
+		  //		     >> seg.epgeominfo[0].lefttrig
+		  //		     >> seg.epgeominfo[0].righttrig
+		       >> seg.epgeominfo[0].dist
+		       >> seg.epgeominfo[1].edgenr
+		  //		     >> seg.epgeominfo[1].lefttrig
+		  //		     >> seg.epgeominfo[1].righttrig
+		       >> seg.epgeominfo[1].dist;
+		// seg.edgenr = seg.epgeominfo[0].edgenr;
+		seg.epgeominfo[0].edgenr = seg.epgeominfo[1].edgenr;
+
+		seg.surfnr1--;
+		seg.surfnr2--;
+	      
+		AddSegment (seg);
+	      }
+	  }
+      
+	if (strcmp (str, "points") == 0)
+	  {
+	    infile >> n;
+	    PrintMessage (3, n, " points");
+	    for (i = 1; i <= n; i++)
+	      {
+		Point3d p;
+		infile >> p.X() >> p.Y() >> p.Z();
+		p.X() *= scale;
+		p.Y() *= scale;
+		p.Z() *= scale;
+		AddPoint (p);
+	      }
+	  }
 
-  for (sei = 0; sei < GetNSE(); sei++)
-    {
-      const Element2d & sel = surfelements[sei];
-      if (sel.IsDeleted()) continue;
+	if (strcmp (str, "identifications") == 0)
+	  {
+	    infile >> n;
+	    for (i = 1; i <= n; i++)
+	      {
+		PointIndex pi1, pi2;
+		int ind;
+		infile >> pi1 >> pi2 >> ind;
+		ident -> Add (pi1, pi2, ind);
+	      }
+	  }
+	if (strcmp (str, "materials") == 0)
+	  {
+	    infile >> n;
+	    for (i = 1; i <= n; i++)
+	      {
+		int nr;
+		string mat;
+		infile >> nr >> mat;
+		SetMaterial (nr, mat.c_str());
+	      }
+	  }
+	
+	if (strcmp (str, "singular_points") == 0)
+	  {
+	    infile >> n;
+	    for (i = 1; i <= n; i++)
+	      {
+		PointIndex pi;
+		infile >> pi;
+		(*this)[pi].SetSingular (1);
+	      }
+	  }
 
-      int si = sel.GetIndex();
-      
-      for (j = 0; j < sel.GetNP(); j++)
-	{
-	  PointIndex pi = sel[j];
-	  bool found = 0;
-	  for (k = 0; k < surfacesonnode[pi].Size(); k++)
-	    if (surfacesonnode[pi][k] == si)
+	if (strcmp (str, "singular_edge_left") == 0)
+	  {
+	    infile >> n;
+	    for (i = 1; i <= n; i++)
 	      {
-		found = 1;
-		break;
+		SegmentIndex si;
+		infile >> si;
+		(*this)[si].singedge_left = 1;
 	      }
-	  
-	  if (!found)
-	    surfacesonnode.Add (pi, si);
+	  }
+	if (strcmp (str, "singular_edge_right") == 0)
+	  {
+	    infile >> n;
+	    for (i = 1; i <= n; i++)
+	      {
+		SegmentIndex si;
+		infile >> si;
+		(*this)[si].singedge_right = 1;
+	      }
+	  }
 
-	  INDEX_2 i2;
-	  i2.I1() = sel.PNumMod(j+1);
-	  i2.I2() = sel.PNumMod(j+2);
-	  i2.Sort();
-	  if (sel.GetNP() <= 4)
-	    boundaryedges->Set (i2, 1);
-	}
-    }
+	if (strcmp (str, "singular_face_inside") == 0)
+	  {
+	    infile >> n;
+	    for (i = 1; i <= n; i++)
+	      {
+		SurfaceElementIndex sei;
+		infile >> sei;
+		GetFaceDescriptor((*this)[sei].GetIndex()).domin_singular = 1;
+	      }
+	  }
 
-  for (sei = 0; sei < GetNSE(); sei++)
-    {
-      const Element2d & sel = surfelements[sei];
-      if (sel.IsDeleted()) continue;
+	if (strcmp (str, "singular_face_outside") == 0)
+	  {
+	    infile >> n;
+	    for (i = 1; i <= n; i++)
+	      {
+		SurfaceElementIndex sei;
+		infile >> sei;
+		GetFaceDescriptor((*this)[sei].GetIndex()).domout_singular = 1;
+	      }
+	  }
 
-      INDEX_3 i3;
-      i3.I1() = sel.PNum(1);
-      i3.I2() = sel.PNum(2);
-      i3.I3() = sel.PNum(3);
-      i3.Sort();
-      surfelementht -> PrepareSet (i3);
-    }
 
-  surfelementht -> AllocateElements();
-  for (sei = 0; sei < GetNSE(); sei++)
-    {
+
+	strcpy (str, "");
+      }
+  
+    CalcSurfacesOfNode ();
+    //  BuildConnectedNodes ();
+    topology -> Update();
+    clusters -> Update();
+  
+    SetNextMajorTimeStamp();
+    //  PrintMemInfo (cout);
+  }
+  
+
+
+
+
+
+  void Mesh :: Merge (const string & filename)
+  {
+    char str[100];
+    int i, n;
+
+    ifstream infile(filename.c_str());
+    if (!infile.good())
+      throw NgException ("mesh file not found");
+
+    int oldnp = GetNP();
+
+    while (infile.good())
+      {
+	infile >> str;
+
+	if (strcmp (str, "surfaceelementsgi") == 0)
+	  {
+	    infile >> n;
+	    PrintMessage (3, n, " surface elements");
+	    for (i = 1; i <= n; i++)
+	      {
+		int j;
+		int surfnr, bcp, domin, domout, nep, faceind = 0;
+		infile >> surfnr >> bcp >> domin >> domout;
+		surfnr--;
+
+		// surfnr = 0;
+		// bcp = 1;
+		domin = 2;
+		domout = 1;
+
+		for (j = 1; j <= facedecoding.Size(); j++)
+		  if (GetFaceDescriptor(j).SurfNr() == surfnr &&
+		      GetFaceDescriptor(j).BCProperty() == bcp &&
+		      GetFaceDescriptor(j).DomainIn() == domin &&
+		      GetFaceDescriptor(j).DomainOut() == domout)
+		    faceind = j;
+
+		if (!faceind)
+		  {
+		    faceind = AddFaceDescriptor (FaceDescriptor(surfnr, domin, domout, 0));
+		    GetFaceDescriptor(faceind).SetBCProperty (bcp);
+		  }
+
+		infile >> nep;
+		if (!nep) nep = 3;
+
+		Element2d tri(nep);
+		tri.SetIndex(faceind);
+
+		for (j = 1; j <= nep; j++)
+		  {
+		    infile >> tri.PNum(j);
+		    tri.PNum(j) = tri.PNum(j) + oldnp;
+		  }
+
+		for (j = 1; j <= nep; j++)
+		  infile >> tri.GeomInfoPi(j).trignum;
+
+		AddSurfaceElement (tri);
+	      }
+	  }
+
+	if (strcmp (str, "points") == 0)
+	  {
+	    infile >> n;
+	    PrintMessage (3, n, " points");
+	    for (i = 1; i <= n; i++)
+	      {
+		Point3d p;
+		infile >> p.X() >> p.Y() >> p.Z();
+		AddPoint (p);
+	      }
+	  }
+
+	strcpy (str, "");
+      }
+  
+    CalcSurfacesOfNode ();
+
+    topology -> Update();
+    clusters -> Update();
+  
+    SetNextMajorTimeStamp();
+  }
+  
+
+
+
+
+
+
+
+   
+
+  bool Mesh :: TestOk () const
+  {
+    for (ElementIndex ei = 0; ei < volelements.Size(); ei++)
+      {
+	for (int j = 0; j < 4; j++)
+	  if ( (*this)[ei][j] <= PointIndex::BASE-1)
+	    {
+	      (*testout) << "El " << ei << " has 0 nodes: ";
+	      for (int k = 0; k < 4; k++)
+		(*testout) << (*this)[ei][k];
+	      break;
+	    }
+      }
+    CheckMesh3D (*this);
+    return 1;
+  }
+
+  void Mesh :: CalcSurfacesOfNode ()
+  {
+    int i, j, k;
+    SurfaceElementIndex sei;
+
+    surfacesonnode.SetSize (GetNP());
+    if (boundaryedges)
+      delete boundaryedges;
+    if (surfelementht)
+      delete surfelementht;
+    if (segmentht)
+      delete segmentht;
+
+    boundaryedges = new INDEX_2_CLOSED_HASHTABLE<int>
+      (3 * (GetNSE() + GetNOpenElements()) + GetNSeg() + 1);
+
+    /*
+      surfelementht = new INDEX_3_HASHTABLE<int> (GetNSE()/4 + 1);
+      segmentht = new INDEX_2_HASHTABLE<int> (GetNSeg() + 1);
+    */
+
+    surfelementht = new INDEX_3_CLOSED_HASHTABLE<int> (3*GetNSE() + 1);
+    segmentht = new INDEX_2_CLOSED_HASHTABLE<int> (3*GetNSeg() + 1);
+
+    for (sei = 0; sei < GetNSE(); sei++)
+      {
+	const Element2d & sel = surfelements[sei];
+	if (sel.IsDeleted()) continue;
+
+	int si = sel.GetIndex();
+      
+	for (j = 0; j < sel.GetNP(); j++)
+	  {
+	    PointIndex pi = sel[j];
+	    bool found = 0;
+	    for (k = 0; k < surfacesonnode[pi].Size(); k++)
+	      if (surfacesonnode[pi][k] == si)
+		{
+		  found = 1;
+		  break;
+		}
+	  
+	    if (!found)
+	      surfacesonnode.Add (pi, si);
+
+	    INDEX_2 i2;
+	    i2.I1() = sel.PNumMod(j+1);
+	    i2.I2() = sel.PNumMod(j+2);
+	    i2.Sort();
+	    if (sel.GetNP() <= 4)
+	      boundaryedges->Set (i2, 1);
+	  }
+      }
+    /*
+      for (sei = 0; sei < GetNSE(); sei++)
+      {
       const Element2d & sel = surfelements[sei];
       if (sel.IsDeleted()) continue;
 
@@ -741,1101 +989,1165 @@ void Mesh :: CalcSurfacesOfNode ()
       i3.I2() = sel.PNum(2);
       i3.I3() = sel.PNum(3);
       i3.Sort();
-      surfelementht -> Set (i3, sel.GetIndex());
-    }
+      surfelementht -> PrepareSet (i3);
+      }
 
-  int np = GetNP();
-  ptyps.SetSize(np);
-  ptyps = INNERPOINT;
+      surfelementht -> AllocateElements();
+    */
+    for (sei = 0; sei < GetNSE(); sei++)
+      {
+	const Element2d & sel = surfelements[sei];
+	if (sel.IsDeleted()) continue;
+
+	INDEX_3 i3;
+	i3.I1() = sel.PNum(1);
+	i3.I2() = sel.PNum(2);
+	i3.I3() = sel.PNum(3);
+	i3.Sort();
+	surfelementht -> Set (i3, sel.GetIndex());
+      }
 
-  if (GetNFD() == 0) //  || GetFaceDescriptor(1).SurfNr() == 0)
-    {
-      for (sei = 0; sei < GetNSE(); sei++)
-	{
-	  const Element2d & sel = surfelements[sei];
-	  if (sel.IsDeleted()) continue;
-	  for (j = 0;  j < sel.GetNP(); j++)
-	    {
-	      PointIndex pi = SurfaceElement(sei)[j];
-	      ptyps[pi] = FIXEDPOINT;
-	    }
-	}
-    }
-  else
-    {
-      for (sei = 0; sei < GetNSE(); sei++)
-	{
-	  const Element2d & sel = surfelements[sei];
-	  if (sel.IsDeleted()) continue;
-	  for (j = 0; j < sel.GetNP(); j++)
-	    {
-	      int pi = sel[j];
-	      int ns = surfacesonnode[pi].Size();
-	      if (ns == 1)
-		ptyps[pi] = SURFACEPOINT;
-	      if (ns == 2)
-		ptyps[pi] = EDGEPOINT;
-	      if (ns >= 3)
-		ptyps[pi] = FIXEDPOINT;
-	    }      
-	}
-    }
+    int np = GetNP();
+    // ptyps.SetSize(np);
+    // ptyps = INNERPOINT;
+    for (PointIndex pi = PointIndex::BASE; 
+	 pi < np+PointIndex::BASE; pi++)
+      points[pi].SetType (INNERPOINT);
 
-  for (i = 0; i < segments.Size(); i++)
-    {
-      const Segment & seg = segments[i];
-      for (j = 1; j <= 2; j++)
-	{
-	  PointIndex hi = (j == 1) ? seg.p1 : seg.p2;
+    if (GetNFD() == 0) //  || GetFaceDescriptor(1).SurfNr() == 0)
+      {
+	for (sei = 0; sei < GetNSE(); sei++)
+	  {
+	    const Element2d & sel = surfelements[sei];
+	    if (sel.IsDeleted()) continue;
+	    for (j = 0;  j < sel.GetNP(); j++)
+	      {
+		PointIndex pi = SurfaceElement(sei)[j];
+		points[pi].SetType(FIXEDPOINT);
+	      }
+	  }
+      }
+    else
+      {
+	for (sei = 0; sei < GetNSE(); sei++)
+	  {
+	    const Element2d & sel = surfelements[sei];
+	    if (sel.IsDeleted()) continue;
+	    for (j = 0; j < sel.GetNP(); j++)
+	      {
+		PointIndex pi = sel[j];
+		int ns = surfacesonnode[pi].Size();
+		if (ns == 1)
+		  points[pi].SetType(SURFACEPOINT);
+		if (ns == 2)
+		  points[pi].SetType(EDGEPOINT);
+		if (ns >= 3)
+		  points[pi].SetType(FIXEDPOINT);
+	      }      
+	  }
+      }
+
+    for (i = 0; i < segments.Size(); i++)
+      {
+	const Segment & seg = segments[i];
+	for (j = 1; j <= 2; j++)
+	  {
+	    PointIndex hi = (j == 1) ? seg.p1 : seg.p2;
 	  
-	  if (ptyps[hi] == INNERPOINT ||
-	      ptyps[hi] == SURFACEPOINT)
-	    ptyps[hi] = EDGEPOINT;
-	}
-    }
+	    if (points[hi].Type() == INNERPOINT ||
+		points[hi].Type() == SURFACEPOINT)
+	      points[hi].SetType(EDGEPOINT);
+	  }
+      }
 
 
-  for (i = 0; i < lockedpoints.Size(); i++)
-    ptyps[lockedpoints[i]] = FIXEDPOINT;
+    for (i = 0; i < lockedpoints.Size(); i++)
+      points[lockedpoints[i]].SetType(FIXEDPOINT);
   
-  for (i = 0; i < openelements.Size(); i++)
-    {
-      const Element2d & sel = openelements[i];
-      for (j = 0; j < sel.GetNP(); j++)
-	{
-	  INDEX_2 i2;
-	  i2.I1() = sel.PNumMod(j+1);
-	  i2.I2() = sel.PNumMod(j+2);
-	  i2.Sort();
-	  boundaryedges->Set (i2, 1);
+    for (i = 0; i < openelements.Size(); i++)
+      {
+	const Element2d & sel = openelements[i];
+	for (j = 0; j < sel.GetNP(); j++)
+	  {
+	    INDEX_2 i2;
+	    i2.I1() = sel.PNumMod(j+1);
+	    i2.I2() = sel.PNumMod(j+2);
+	    i2.Sort();
+	    boundaryedges->Set (i2, 1);
 
-	  ptyps[sel[j]] = FIXEDPOINT;
-	}
-    }
+	    points[sel[j]].SetType(FIXEDPOINT);
+	  }
+      }
 
-  eltyps.SetSize (GetNE());
-  eltyps = FREEELEMENT;
+    eltyps.SetSize (GetNE());
+    eltyps = FREEELEMENT;
 
-  for (i = 0; i < GetNSeg(); i++)
-    {
-      const Segment & seg = segments[i];
-      INDEX_2 i2(seg.p1, seg.p2);
-      i2.Sort();
+    for (i = 0; i < GetNSeg(); i++)
+      {
+	const Segment & seg = segments[i];
+	INDEX_2 i2(seg.p1, seg.p2);
+	i2.Sort();
 
-      boundaryedges -> Set (i2, 2);
-      segmentht -> Set (i2, 1);
-    }
-}
+	boundaryedges -> Set (i2, 2);
+	segmentht -> Set (i2, i);
+      }
+  }
 
 
-void Mesh :: FixPoints (const BitArray & fixpoints)
-{
-  if (fixpoints.Size() != GetNP())
-    {
-      cerr << "Mesh::FixPoints: sizes don´t fit" << endl;
-      return;
-    }
-  int np = GetNP();
-  for (int i = 1; i <= np; i++)
-    if (fixpoints.Test(i))
+  void Mesh :: FixPoints (const BitArray & fixpoints)
+  {
+    if (fixpoints.Size() != GetNP())
       {
-	ptyps.Elem(i) = FIXEDPOINT;
+	cerr << "Mesh::FixPoints: sizes don´t fit" << endl;
+	return;
       }
-}
+    int np = GetNP();
+    for (int i = 1; i <= np; i++)
+      if (fixpoints.Test(i))
+	{
+	  points.Elem(i).SetType (FIXEDPOINT);
+	}
+  }
 
 
-void Mesh :: FindOpenElements (int dom)
-{
-  int i, ii, j, k, l;
-  PointIndex pi;
-  SurfaceElementIndex sei;
-  Element2d hel;
+  void Mesh :: FindOpenElements (int dom)
+  {
+    int i, ii, j, k, l;
+    PointIndex pi;
+    SurfaceElementIndex sei;
+    Element2d hel;
 
 
+    if (1)   
+      { // nodebased 
 
-  if (1)   
-    { // nodebased 
+	int np = GetNP();
+	int ne = GetNE();
+	int nse = GetNSE();
 
-      int np = GetNP();
-      int ne = GetNE();
-      int nse = GetNSE();
+	ARRAY<int,PointIndex::BASE> numonpoint(np);
 
-      ARRAY<int,PointIndex::BASE> numonpoint(np);
+	Element2d hel;
 
-      Element2d hel;
+	numonpoint = 0;
+	ElementIndex ei;
+	for (ei = 0; ei < ne; ei++)
+	  {
+	    const Element & el = (*this)[ei];
+	    if (dom == 0 || dom == el.GetIndex())
+	      {
+		if (el.GetNP() == 4)
+		  {
+		    INDEX_4 i4(el[0], el[1], el[2], el[3]);
+		    i4.Sort();
+		    numonpoint[i4.I1()]++;
+		    numonpoint[i4.I2()]++;
+		  }
+		else
+		  for (j = 0; j < el.GetNP(); j++)
+		    numonpoint[el[j]]++;
+	      }
+	  }
 
-      numonpoint = 0;
-      /*
-      for (i = 1; i <= np; i++)
-	numonpoint.Elem(i) = 0;
-      */
-      ElementIndex ei;
-      for (ei = 0; ei < ne; ei++)
-	{
-	  const Element & el = (*this)[ei];
-	  if (el.GetNP() == 4)
-	    {
-	      INDEX_4 i4(el[0], el[1], el[2], el[3]);
-	      i4.Sort();
-	      numonpoint[i4.I1()]++;
-	      numonpoint[i4.I2()]++;
-	    }
-	  else
-	    for (j = 0; j < el.GetNP(); j++)
-	      numonpoint[el[j]]++;
-	}
-      TABLE<ElementIndex,PointIndex::BASE> elsonpoint(numonpoint);
-      for (ei = 0; ei < ne; ei++)
-	{
-	  const Element & el = (*this)[ei];
-	  if (el.GetNP() == 4)
-	    {
-	      INDEX_4 i4(el[0], el[1], el[2], el[3]);
-	      i4.Sort();
-	      elsonpoint.Add (i4.I1(), ei);
-	      elsonpoint.Add (i4.I2(), ei);
-	    }
-	  else
-	    for (j = 0; j < el.GetNP(); j++)
-	      elsonpoint.Add (el[j], ei);
-	}
+	TABLE<ElementIndex,PointIndex::BASE> elsonpoint(numonpoint);
+	for (ei = 0; ei < ne; ei++)
+	  {
+	    const Element & el = (*this)[ei];
+	    if (dom == 0 || dom == el.GetIndex())
+	      {
+		if (el.GetNP() == 4)
+		  {
+		    INDEX_4 i4(el[0], el[1], el[2], el[3]);
+		    i4.Sort();
+		    elsonpoint.Add (i4.I1(), ei);
+		    elsonpoint.Add (i4.I2(), ei);
+		  }
+		else
+		  for (j = 0; j < el.GetNP(); j++)
+		    elsonpoint.Add (el[j], ei);
+	      }
+	  }
 
-      numonpoint = 0;
-      for (i = 0; i < nse; i++)
-	{
-	  hel = surfelements[i];
-	  hel.NormalizeNumbering();	  
-	  numonpoint[hel[0]]++;
-	}
+	numonpoint = 0;
+	for (i = 0; i < nse; i++)
+	  {
+	    int ind = surfelements[i].GetIndex();
+	    if (
+		GetFaceDescriptor(ind).DomainIn() && 
+		(dom == 0 || dom == GetFaceDescriptor(ind).DomainIn())
+		||
+		GetFaceDescriptor(ind).DomainOut() && 
+		(dom == 0 || dom == GetFaceDescriptor(ind).DomainOut())
+		)
+	      {
+		hel = surfelements[i];
+		hel.NormalizeNumbering();	  
+		numonpoint[hel[0]]++;
+	      }
+	  }
 
-      TABLE<SurfaceElementIndex,PointIndex::BASE> selsonpoint(numonpoint);
-      for (i = 0; i < nse; i++)
-	{
-	  hel = surfelements[i];
-	  hel.NormalizeNumbering();	  
-	  selsonpoint.Add (hel[0], i);
-	}
+	TABLE<SurfaceElementIndex,PointIndex::BASE> selsonpoint(numonpoint);
+	for (i = 0; i < nse; i++)
+	  {
+	    int ind = surfelements[i].GetIndex();
+	    if (
+		GetFaceDescriptor(ind).DomainIn() && 
+		(dom == 0 || dom == GetFaceDescriptor(ind).DomainIn())
+		||
+		GetFaceDescriptor(ind).DomainOut() && 
+		(dom == 0 || dom == GetFaceDescriptor(ind).DomainOut())
+		)
+	      {
+		hel = surfelements[i];
+		hel.NormalizeNumbering();	  
+		selsonpoint.Add (hel[0], i);
+	      }
+	  }
 
 
-      INDEX_3_CLOSED_HASHTABLE<INDEX_2> faceht(100);   
-      openelements.SetSize(0);
+	INDEX_3_CLOSED_HASHTABLE<INDEX_2> faceht(100);   
+	openelements.SetSize(0);
       
-      for (pi = PointIndex::BASE; 
-	   pi < np+PointIndex::BASE; pi++)
-	{
-	  faceht.SetSize (2 * selsonpoint[pi].Size() + 4 * elsonpoint[pi].Size());
-
-
-	  FlatArray<SurfaceElementIndex> row = selsonpoint[pi];
-	  for (ii = 0; ii < row.Size(); ii++)
+	for (pi = PointIndex::BASE; 
+	     pi < np+PointIndex::BASE; pi++)
+	  if (selsonpoint[pi].Size()+elsonpoint[pi].Size())
 	    {
-	      hel = SurfaceElement(row[ii]);
-	      int ind = hel.GetIndex();	  
-  
-	      if (GetFaceDescriptor(ind).DomainIn() && 
-		  (dom == 0 || dom == GetFaceDescriptor(ind).DomainIn()) )
+	      faceht.SetSize (2 * selsonpoint[pi].Size() + 4 * elsonpoint[pi].Size());
+
+	      FlatArray<SurfaceElementIndex> row = selsonpoint[pi];
+	      for (ii = 0; ii < row.Size(); ii++)
 		{
-		  hel.NormalizeNumbering();
-		  if (hel.PNum(1) == pi)
+		  hel = SurfaceElement(row[ii]);
+		  int ind = hel.GetIndex();	  
+  
+		  if (GetFaceDescriptor(ind).DomainIn() && 
+		      (dom == 0 || dom == GetFaceDescriptor(ind).DomainIn()) )
 		    {
-		      INDEX_3 i3(hel[0], hel[1], hel[2]);
-		      INDEX_2 i2 (GetFaceDescriptor(ind).DomainIn(), 
-				  (hel.GetNP() == 3) 
-				  ? PointIndex (PointIndex::BASE-1)
-				  : hel.PNum(4));
-		      faceht.Set (i3, i2);
+		      hel.NormalizeNumbering();
+		      if (hel.PNum(1) == pi)
+			{
+			  INDEX_3 i3(hel[0], hel[1], hel[2]);
+			  INDEX_2 i2 (GetFaceDescriptor(ind).DomainIn(), 
+				      (hel.GetNP() == 3) 
+				      ? PointIndex (PointIndex::BASE-1)
+				      : hel.PNum(4));
+			  faceht.Set (i3, i2);
+			}
 		    }
-		}
-	      if (GetFaceDescriptor(ind).DomainOut() &&
-		  (dom == 0 || dom == GetFaceDescriptor(ind).DomainOut()) )
-		{
-		  hel.Invert();
-		  hel.NormalizeNumbering();
-		  if (hel.PNum(1) == pi)
+		  if (GetFaceDescriptor(ind).DomainOut() &&
+		      (dom == 0 || dom == GetFaceDescriptor(ind).DomainOut()) )
 		    {
-		      INDEX_3 i3(hel[0], hel[1], hel[2]);
-		      INDEX_2 i2 (GetFaceDescriptor(ind).DomainOut(), 
-				  (hel.GetNP() == 3) 
-				  ? PointIndex (PointIndex::BASE-1)
-				  : hel.PNum(4));
-		      faceht.Set (i3, i2);
+		      hel.Invert();
+		      hel.NormalizeNumbering();
+		      if (hel.PNum(1) == pi)
+			{
+			  INDEX_3 i3(hel[0], hel[1], hel[2]);
+			  INDEX_2 i2 (GetFaceDescriptor(ind).DomainOut(), 
+				      (hel.GetNP() == 3) 
+				      ? PointIndex (PointIndex::BASE-1)
+				      : hel.PNum(4));
+			  faceht.Set (i3, i2);
+			}
 		    }
 		}
-	    }
 
 	  
-	  FlatArray<ElementIndex> rowel = elsonpoint[pi];
-	  for (ii = 0; ii < rowel.Size(); ii++)
-	    {
-	      const Element & el = VolumeElement(rowel[ii]);
-
-	      if (dom == 0 || el.GetIndex() == dom)
+	      FlatArray<ElementIndex> rowel = elsonpoint[pi];
+	      for (ii = 0; ii < rowel.Size(); ii++)
 		{
-		  for (j = 1; j <= el.GetNFaces(); j++)
-		    {
-		      el.GetFace (j, hel);
-		      hel.Invert();
-		      hel.NormalizeNumbering();
+		  const Element & el = VolumeElement(rowel[ii]);
 
-		      if (hel[0] == pi)
+		  if (dom == 0 || el.GetIndex() == dom)
+		    {
+		      for (j = 1; j <= el.GetNFaces(); j++)
 			{
-			  INDEX_3 i3(hel[0], hel[1], hel[2]);
-			  
-			  if (faceht.Used (i3))
+			  el.GetFace (j, hel);
+			  hel.Invert();
+			  hel.NormalizeNumbering();
+
+			  if (hel[0] == pi)
 			    {
-			      INDEX_2 i2 = faceht.Get(i3);
-			      if (i2.I1() == el.GetIndex())
+			      INDEX_3 i3(hel[0], hel[1], hel[2]);
+			  
+			      if (faceht.Used (i3))
 				{
-				  i2.I1() = PointIndex::BASE-1;
-				  faceht.Set (i3, i2);
+				  INDEX_2 i2 = faceht.Get(i3);
+				  if (i2.I1() == el.GetIndex())
+				    {
+				      i2.I1() = PointIndex::BASE-1;
+				      faceht.Set (i3, i2);
+				    }
+				  else
+				    {
+				      if (i2.I1() == 0)
+					{
+					  PrintSysError ("more elements on face");
+					  (*testout)  << "more elements on face!!!" << endl;
+					  (*testout) << "el = " << el << endl;
+					  (*testout) << "hel = " << hel << endl;
+					  (*testout) << "face = " << i3 << endl;
+					  (*testout) << "points = " << endl;
+					  for (int jj = 1; jj <= 3; jj++)
+					    (*testout) << "p = " << Point(i3.I(jj)) << endl;
+					}
+				    }
 				}
 			      else
 				{
-				  if (i2.I1() == 0)
-				    {
-				      PrintSysError ("more elements on face");
-				      (*testout)  << "more elements on face!!!" << endl;
-				      (*testout) << "el = " << el << endl;
-				      (*testout) << "hel = " << hel << endl;
-				      (*testout) << "face = " << i3 << endl;
-				      (*testout) << "points = " << endl;
-				      for (int jj = 1; jj <= 3; jj++)
-					(*testout) << "p = " << Point(i3.I(jj)) << endl;
-				    }
+				  hel.Invert();
+				  hel.NormalizeNumbering();
+				  INDEX_3 i3(hel[0], hel[1], hel[2]);
+				  INDEX_2 i2(el.GetIndex(), 
+					     (hel.GetNP() == 3) 
+					     ? PointIndex (PointIndex::BASE-1)
+					     : hel[3]);
+				  faceht.Set (i3, i2);
 				}
 			    }
-			  else
-			    {
-			      hel.Invert();
-			      hel.NormalizeNumbering();
-			      INDEX_3 i3(hel[0], hel[1], hel[2]);
-			      INDEX_2 i2(el.GetIndex(), 
-					 (hel.GetNP() == 3) 
-					 ? PointIndex (PointIndex::BASE-1)
-					 : hel[3]);
-			      faceht.Set (i3, i2);
-			    }
 			}
 		    }
 		}
-	    }
 
-	  for (i = 1; i <= faceht.Size(); i++)
-	    if (faceht.UsedPos (i))
-	      {
-		INDEX_3 i3;
-		INDEX_2 i2;
-		faceht.GetData (i, i3, i2);
-		if (i2.I1() != PointIndex::BASE-1)
+	      for (i = 1; i <= faceht.Size(); i++)
+		if (faceht.UsedPos (i))
 		  {
-		    Element2d tri;
-		    tri.SetType ( (i2.I2() == PointIndex::BASE-1) ? TRIG : QUAD);
-		    for (l = 0; l < 3; l++)
-		      tri[l] = i3.I(l+1);
-		    tri.PNum(4) = i2.I2();
-		    tri.SetIndex (i2.I1());
-
-		    //	tri.Invert();
+		    INDEX_3 i3;
+		    INDEX_2 i2;
+		    faceht.GetData (i, i3, i2);
+		    if (i2.I1() != PointIndex::BASE-1)
+		      {
+			Element2d tri;
+			tri.SetType ( (i2.I2() == PointIndex::BASE-1) ? TRIG : QUAD);
+			for (l = 0; l < 3; l++)
+			  tri[l] = i3.I(l+1);
+			tri.PNum(4) = i2.I2();
+			tri.SetIndex (i2.I1());
+
+			//	tri.Invert();
 		    
-		    openelements.Append (tri);
+			openelements.Append (tri);
+		      }
 		  }
-	      }
-	}
-    }
+	    }
+      }
 
-  else if (GetNE() || 1)
-    {
-      // new version, general elemetns
-      // hash index: pnum1-3
-      // hash data : domain nr, pnum4
 
-      openelements.SetSize(0);
+#ifdef OLD
+
+    else if (GetNE() || 1)
+      {
+	// new version, general elemetns
+	// hash index: pnum1-3
+	// hash data : domain nr, pnum4
+
+	openelements.SetSize(0);
       
 
-      const int steps = 4;
+	const int steps = 4;
 
-      for (k = 0; k < steps; k++)
-	{
+	for (k = 0; k < steps; k++)
+	  {
 	  
-	  INDEX_3_CLOSED_HASHTABLE<INDEX_2> faceht( (5 * GetNE() + 2 * GetNSE() ) / steps +1);   
+	    INDEX_3_CLOSED_HASHTABLE<INDEX_2> faceht( (5 * GetNE() + 2 * GetNSE() ) / steps +1);   
 	  
-	  for (i = 1; i <= GetNSE(); i++)
-	    {
-	      hel = SurfaceElement(i);
-	      int ind = hel.GetIndex();	  
+	    for (i = 1; i <= GetNSE(); i++)
+	      {
+		hel = SurfaceElement(i);
+		int ind = hel.GetIndex();	  
 	      
 	      
-	      if (GetFaceDescriptor(ind).DomainIn() && 
-		  (dom == 0 || dom == GetFaceDescriptor(ind).DomainIn()) )
-		{
-		  hel.Invert();
-		  hel.NormalizeNumbering();
-		  if (hel.PNum(1) % steps == k)
-		    {
-		      INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
-		      INDEX_2 i2 (GetFaceDescriptor(ind).DomainIn(), 
-				  (hel.GetNP() == 3) 
-				  ? PointIndex (PointIndex::BASE-1)
-				  : hel.PNum(4));
-		      faceht.Set (i3, i2);
-		    }
-		}
-	      if (GetFaceDescriptor(ind).DomainOut() &&
-		  (dom == 0 || dom == GetFaceDescriptor(ind).DomainOut()) )
-		{
-		  hel.NormalizeNumbering();
-		  if (hel.PNum(1) % steps == k)
-		    {
-		      INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
-		      INDEX_2 i2 (GetFaceDescriptor(ind).DomainOut(), 
-				  (hel.GetNP() == 3) 
-				  ? PointIndex(0)
-				  : hel.PNum(4));
-		      faceht.Set (i3, i2);
-		    }
-		}
-	    }
+		if (GetFaceDescriptor(ind).DomainIn() && 
+		    (dom == 0 || dom == GetFaceDescriptor(ind).DomainIn()) )
+		  {
+		    hel.Invert();
+		    hel.NormalizeNumbering();
+		    if (hel.PNum(1) % steps == k)
+		      {
+			INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
+			INDEX_2 i2 (GetFaceDescriptor(ind).DomainIn(), 
+				    (hel.GetNP() == 3) 
+				    ? PointIndex (PointIndex::BASE-1)
+				    : hel.PNum(4));
+			faceht.Set (i3, i2);
+		      }
+		  }
+		if (GetFaceDescriptor(ind).DomainOut() &&
+		    (dom == 0 || dom == GetFaceDescriptor(ind).DomainOut()) )
+		  {
+		    hel.NormalizeNumbering();
+		    if (hel.PNum(1) % steps == k)
+		      {
+			INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
+			INDEX_2 i2 (GetFaceDescriptor(ind).DomainOut(), 
+				    (hel.GetNP() == 3) 
+				    ? PointIndex(0)
+				    : hel.PNum(4));
+			faceht.Set (i3, i2);
+		      }
+		  }
+	      }
 
-	  for (i = 1; i <= GetNE(); i++)
-	    {
-	      const Element & el = VolumeElement(i);
+	    for (i = 1; i <= GetNE(); i++)
+	      {
+		const Element & el = VolumeElement(i);
 	      
-	      if (dom == 0 || el.GetIndex() == dom)
-		{
-		  for (j = 1; j <= el.GetNFaces(); j++)
-		    {
-		      el.GetFace (j, hel);
+		if (dom == 0 || el.GetIndex() == dom)
+		  {
+		    for (j = 1; j <= el.GetNFaces(); j++)
+		      {
+			el.GetFace (j, hel);
 		      
-		      hel.NormalizeNumbering();
-		      if (hel.PNum(1) % steps != k)
-			continue;
+			hel.NormalizeNumbering();
+			if (hel.PNum(1) % steps != k)
+			  continue;
 		      
-		      INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
-
-		      int pos = faceht.Position (i3);
-		      if (pos)
-			{
-			  INDEX_2 i2;
-			  faceht.GetData (pos, i2);
-			  if (i2.I1() == el.GetIndex())
-			    {
-			      i2.I1() = 0;
-			      faceht.SetData (pos, i2);
-			    }
-			  else
-			    {
-			      if (i2.I1() == 0)
-				PrintSysError ("more elements on face");
-			    }
-			}
-		      else
-			{
-			  hel.Invert();
-			  hel.NormalizeNumbering();
-			  INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
-			  INDEX_2 i2(el.GetIndex(), 
-				     (hel.GetNP() == 3) 
-				     ? PointIndex(0)
-				     : hel.PNum(4));
-			  faceht.Set (i3, i2);
-			}
-		    }
-		}
-	    }  
+			INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
+
+			int pos = faceht.Position (i3);
+			if (pos)
+			  {
+			    INDEX_2 i2;
+			    faceht.GetData (pos, i2);
+			    if (i2.I1() == el.GetIndex())
+			      {
+				i2.I1() = 0;
+				faceht.SetData (pos, i2);
+			      }
+			    else
+			      {
+				if (i2.I1() == 0)
+				  PrintSysError ("more elements on face");
+			      }
+			  }
+			else
+			  {
+			    hel.Invert();
+			    hel.NormalizeNumbering();
+			    INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
+			    INDEX_2 i2(el.GetIndex(), 
+				       (hel.GetNP() == 3) 
+				       ? PointIndex(0)
+				       : hel.PNum(4));
+			    faceht.Set (i3, i2);
+			  }
+		      }
+		  }
+	      }  
 	  
-	  for (i = 1; i <= faceht.Size(); i++)
-	    if (faceht.UsedPos (i))
-	      {
-		INDEX_3 i3;
-		INDEX_2 i2;
-		faceht.GetData (i, i3, i2);
-		if (i2.I1() != 0)
-		  {
-		    Element2d tri;
-		    tri.SetType ( (i2.I2() == 0) ? TRIG : QUAD);
-		    for (l = 1; l <= 3; l++)
-		      tri.PNum(l) = i3.I(l);
-		    tri.PNum(4) = i2.I2();
-		    tri.SetIndex (i2.I1());
-
-		    tri.Invert();
+	    for (i = 1; i <= faceht.Size(); i++)
+	      if (faceht.UsedPos (i))
+		{
+		  INDEX_3 i3;
+		  INDEX_2 i2;
+		  faceht.GetData (i, i3, i2);
+		  if (i2.I1() != 0)
+		    {
+		      Element2d tri;
+		      tri.SetType ( (i2.I2() == 0) ? TRIG : QUAD);
+		      for (l = 1; l <= 3; l++)
+			tri.PNum(l) = i3.I(l);
+		      tri.PNum(4) = i2.I2();
+		      tri.SetIndex (i2.I1());
+
+		      tri.Invert();
 		    
-		    openelements.Append (tri);
-		  }
-	      }
-	  /*
-	  cout << "FindOpenElements, mem = ";
-	  faceht.PrintMemInfo (cout);
-	  */
-	}
+		      openelements.Append (tri);
+		    }
+		}
+	    /*
+	      cout << "FindOpenElements, mem = ";
+	      faceht.PrintMemInfo (cout);
+	    */
+	  }
 
 
 
 
-      /*
+	/*
 	// open hashing version:
 
 	INDEX_3_HASHTABLE<INDEX_2> faceht(4 * GetNE()+GetNSE()+1);   
  
-      for (i = 1; i <= GetNSE(); i++)
+	for (i = 1; i <= GetNSE(); i++)
 	{
-	  Element2d hel = SurfaceElement(i);
-	  int ind = hel.GetIndex();	  
+	Element2d hel = SurfaceElement(i);
+	int ind = hel.GetIndex();	  
 	  
 
-	  if (GetFaceDescriptor(ind).DomainIn() && 
-	      (dom == 0 || dom == GetFaceDescriptor(ind).DomainIn()) )
-	    {
-	      hel.NormalizeNumbering();
-	      INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
-	      INDEX_2 i2 (GetFaceDescriptor(ind).DomainIn(), 
-			  (hel.GetNP() == 3) ? 0 : hel.PNum(4));
-	      faceht.Set (i3, i2);
-	    }
-	  if (GetFaceDescriptor(ind).DomainOut() &&
-	      (dom == 0 || dom == GetFaceDescriptor(ind).DomainOut()) )
-	    {
-	      hel.Invert();
-	      hel.NormalizeNumbering();
-	      INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
-	      INDEX_2 i2 (GetFaceDescriptor(ind).DomainOut(), 
-			  (hel.GetNP() == 3) ? 0 : hel.PNum(4));
-	      faceht.Set (i3, i2);
-	    }
+	if (GetFaceDescriptor(ind).DomainIn() && 
+	(dom == 0 || dom == GetFaceDescriptor(ind).DomainIn()) )
+	{
+	hel.NormalizeNumbering();
+	INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
+	INDEX_2 i2 (GetFaceDescriptor(ind).DomainIn(), 
+	(hel.GetNP() == 3) ? 0 : hel.PNum(4));
+	faceht.Set (i3, i2);
+	}
+	if (GetFaceDescriptor(ind).DomainOut() &&
+	(dom == 0 || dom == GetFaceDescriptor(ind).DomainOut()) )
+	{
+	hel.Invert();
+	hel.NormalizeNumbering();
+	INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
+	INDEX_2 i2 (GetFaceDescriptor(ind).DomainOut(), 
+	(hel.GetNP() == 3) ? 0 : hel.PNum(4));
+	faceht.Set (i3, i2);
+	}
 	}
 
-      for (i = 1; i <= GetNE(); i++)
+	for (i = 1; i <= GetNE(); i++)
 	{
-	  const Element & el = VolumeElement(i);
-	  //	  INDEX_3 i3;
+	const Element & el = VolumeElement(i);
+	//	  INDEX_3 i3;
 
-	  if (dom == 0 || el.GetIndex() == dom)
-	    {
-	      for (j = 1; j <= el.GetNFaces(); j++)
-		{
-		  Element2d hel;
-		  el.GetFace (j, hel);
-		  hel.Invert();
+	if (dom == 0 || el.GetIndex() == dom)
+	{
+	for (j = 1; j <= el.GetNFaces(); j++)
+	{
+	Element2d hel;
+	el.GetFace (j, hel);
+	hel.Invert();
 		  
-		  hel.NormalizeNumbering();
-		  INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
+	hel.NormalizeNumbering();
+	INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
 		  
-		  if (faceht.Used (i3))
-		    {
-		      INDEX_2 i2 = faceht.Get(i3);
-		      if (i2.I1() == el.GetIndex())
-			{
-			  i2.I1() = 0;
-			  faceht.Set (i3, i2);
-			}
-		      else
-			{
-			  if (i2.I1() == 0)
-			    PrintSysError ("more elements on face");
-			}
-		    }
-		  else
-		    {
-		      hel.Invert();
-		      hel.NormalizeNumbering();
-		      INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
-		      INDEX_2 i2(el.GetIndex(), 
-				 (hel.GetNP() == 3) ? 0 : hel.PNum(4));
-		      faceht.Set (i3, i2);
-		    }
-		}
-	    }
+	if (faceht.Used (i3))
+	{
+	INDEX_2 i2 = faceht.Get(i3);
+	if (i2.I1() == el.GetIndex())
+	{
+	i2.I1() = 0;
+	faceht.Set (i3, i2);
+	}
+	else
+	{
+	if (i2.I1() == 0)
+	PrintSysError ("more elements on face");
+	}
+	}
+	else
+	{
+	hel.Invert();
+	hel.NormalizeNumbering();
+	INDEX_3 i3(hel.PNum(1), hel.PNum(2), hel.PNum(3));
+	INDEX_2 i2(el.GetIndex(), 
+	(hel.GetNP() == 3) ? 0 : hel.PNum(4));
+	faceht.Set (i3, i2);
+	}
+	}
+	}
 	}  
 
       
-      openelements.SetSize(0);
-      for (i = 1; i <= faceht.GetNBags(); i++)
+	openelements.SetSize(0);
+	for (i = 1; i <= faceht.GetNBags(); i++)
 	for (j = 1; j <= faceht.GetBagSize(i); j++)
+	{
+	INDEX_3 i3;
+	INDEX_2 i2;
+	faceht.GetData (i, j, i3, i2);
+	if (i2.I1() != 0)
+	{
+	Element2d tri;
+	tri.SetNP ( (i2.I2() == 0) ? 3 : 4);
+	for (k = 1; k <= 3; k++)
+	tri.PNum(k) = i3.I(k);
+	tri.PNum(4) = i2.I2();
+	tri.SetIndex (i2.I1());
+
+	openelements.Append (tri);
+	}
+	}
+	*/
+
+      }
+    else 
+      {
+	for (i = 1; i <= GetNSE(); i++)
 	  {
-	    INDEX_3 i3;
-	    INDEX_2 i2;
-	    faceht.GetData (i, j, i3, i2);
-	    if (i2.I1() != 0)
-	      {
-		Element2d tri;
-		tri.SetNP ( (i2.I2() == 0) ? 3 : 4);
-		for (k = 1; k <= 3; k++)
-		  tri.PNum(k) = i3.I(k);
-		tri.PNum(4) = i2.I2();
-		tri.SetIndex (i2.I1());
+	    Element2d hel = SurfaceElement(i);
+	    int ind = SurfaceElement(i).GetIndex();
 
-		openelements.Append (tri);
+	    if (GetFaceDescriptor(ind).DomainIn())
+	      openelements.Append (hel);
+	    if (GetFaceDescriptor(ind).DomainOut())
+	      {
+		hel.Invert();
+		openelements.Append (hel);
 	      }
 	  }
-      */
+      }
+#endif
 
-    }
-  else 
-    {
-      for (i = 1; i <= GetNSE(); i++)
-	{
-	  Element2d hel = SurfaceElement(i);
-	  int ind = SurfaceElement(i).GetIndex();
 
-	  if (GetFaceDescriptor(ind).DomainIn())
-	    openelements.Append (hel);
-	  if (GetFaceDescriptor(ind).DomainOut())
-	    {
-	      hel.Invert();
-	      openelements.Append (hel);
-	    }
-	}
-    }
+    int cnt3 = 0, cnt4 = 0;
+    for (i = 1; i <= openelements.Size(); i++)
+      if (openelements.Elem(i).GetNP() == 3)
+	cnt3++;
+      else
+	cnt4++;
 
 
-  int cnt3 = 0, cnt4 = 0;
-  for (i = 1; i <= openelements.Size(); i++)
-    if (openelements.Elem(i).GetNP() == 3)
-      cnt3++;
-    else
-      cnt4++;
+    MyStr treequad;
+    if (cnt4)
+      treequad = MyStr(" (") + MyStr(cnt3) + MyStr (" + ") + 
+	MyStr(cnt4) + MyStr(")");
 
+    PrintMessage (5, openelements.Size(), treequad, " open elements");
+
+
+    for (i = 1; i <= openelements.Size(); i++)
+      {
+	const Element2d & sel = openelements.Get(i);
 
-  MyStr treequad;
-  if (cnt4)
-    treequad = MyStr(" (") + MyStr(cnt3) + MyStr (" + ") + 
-      MyStr(cnt4) + MyStr(")");
+	if (boundaryedges)
+	  for (j = 1; j <= sel.GetNP(); j++)
+	    {
+	      INDEX_2 i2;
+	      i2.I1() = sel.PNumMod(j);
+	      i2.I2() = sel.PNumMod(j+1);
+	      i2.Sort();
+	      boundaryedges->Set (i2, 1);
+	    }
+      
+	for (j = 1; j <= 3; j++)
+	  {
+	    int pi = sel.PNum(j);
+	    if (pi < points.Size()+PointIndex::BASE)
+	      points[pi].SetType (FIXEDPOINT);
+	  }
+      }
 
-  PrintMessage (5, openelements.Size(), treequad, " open elements");
 
+    for (i = 1; i <= GetNSeg(); i++)
+      {
+	const Segment & seg = LineSegment(i);
+	INDEX_2 i2(seg.p1, seg.p2);
+	i2.Sort();
 
-  for (i = 1; i <= openelements.Size(); i++)
-    {
-      const Element2d & sel = openelements.Get(i);
+	if (!boundaryedges->Used (i2))
+	  cerr << "WARNING: no boundedge, but seg edge: " << i2 << endl;
 
-      if (boundaryedges)
-	for (j = 1; j <= sel.GetNP(); j++)
-	  {
-	    INDEX_2 i2;
-	    i2.I1() = sel.PNumMod(j);
-	    i2.I2() = sel.PNumMod(j+1);
-	    i2.Sort();
-	    boundaryedges->Set (i2, 1);
-	  }
-      
-      for (j = 1; j <= 3; j++)
-	{
-	  int pi = sel.PNum(j);
-	  if (pi < ptyps.Size()+PointIndex::BASE)
-	    ptyps[pi] = FIXEDPOINT;
-	}
-    }
+	boundaryedges -> Set (i2, 2);
+	segmentht -> Set (i2, i-1);
+      }
+  }
 
+  int Mesh :: HasOpenQuads () const
+  {
+    int i;
+    int no = GetNOpenElements();
+    for (i = 1; i <= no; i++)
+      if (OpenElement(i).GetNP() == 4)
+	return 1;
+    return 0;
+  }
 
-  for (i = 1; i <= GetNSeg(); i++)
-    {
-      const Segment & seg = LineSegment(i);
-      INDEX_2 i2(seg.p1, seg.p2);
-      i2.Sort();
 
-      if (!boundaryedges->Used (i2))
-	cerr << "WARNING: no boundedge, but seg edge: " << i2 << endl;
 
-      boundaryedges -> Set (i2, 2);
-      segmentht -> Set (i2, 1);
-    }
-}
 
-int Mesh :: HasOpenQuads () const
-{
-  int i;
-  int no = GetNOpenElements();
-  for (i = 1; i <= no; i++)
-    if (OpenElement(i).GetNP() == 4)
-      return 1;
-  return 0;
-}
 
+  void Mesh :: FindOpenSegments (int surfnr)
+  {
+    int i, j, k;
 
+    // new version, general elemetns
+    // hash index: pnum1-2
+    // hash data : surfnr,  surfel-nr (pos) or segment nr(neg)
+    INDEX_2_HASHTABLE<INDEX_2> faceht(4 * GetNSE()+GetNSeg()+1);   
+  
+    PrintMessage (5, "Test Opensegments");
+    for (i = 1; i <= GetNSeg(); i++)
+      {
+	const Segment & seg = LineSegment (i);
 
+	if (surfnr == 0 || seg.si == surfnr)
+	  {
+	    INDEX_2 key(seg.p1, seg.p2);
+	    INDEX_2 data(seg.si, -i);
 
+	    if (faceht.Used (key))
+	      {
+		cerr << "ERROR: Segment " << seg << " already used" << endl;
+		(*testout) << "ERROR: Segment " << seg << " already used" << endl;
+	      }
 
-void Mesh :: FindOpenSegments (int surfnr)
-{
-  int i, j, k;
+	    faceht.Set (key, data);
+	  }
+      }
 
-  // new version, general elemetns
-  // hash index: pnum1-2
-  // hash data : surfnr,  surfel-nr (pos) or segment nr(neg)
-  INDEX_2_HASHTABLE<INDEX_2> faceht(4 * GetNSE()+GetNSeg()+1);   
-  
-  PrintMessage (5, "Test Opensegments");
-  for (i = 1; i <= GetNSeg(); i++)
-    {
-      const Segment & seg = LineSegment (i);
 
-      if (surfnr == 0 || seg.si == surfnr)
-	{
-	  INDEX_2 key(seg.p1, seg.p2);
-	  INDEX_2 data(seg.si, -i);
+    for (i = 1; i <= GetNSeg(); i++)
+      {
+	const Segment & seg = LineSegment (i);
 
-	  if (faceht.Used (key))
-	    {
-	      cerr << "ERROR: Segment " << seg << " already used" << endl;
-	      (*testout) << "ERROR: Segment " << seg << " already used" << endl;
-	    }
+	if (surfnr == 0 || seg.si == surfnr)
+	  {
+	    INDEX_2 key(seg.p2, seg.p1);
+	    if (!faceht.Used(key))
+	      {
+		cerr << "ERROR: Segment " << seg << " brother not used" << endl;
+		(*testout) << "ERROR: Segment " << seg << " brother not used" << endl;
+	      }
+	  }
+      }
 
-	  faceht.Set (key, data);
-	}
-    }
 
+    for (i = 1; i <= GetNSE(); i++)
+      {
+	const Element2d & el = SurfaceElement(i);
+	if (el.IsDeleted()) continue;
 
-  for (i = 1; i <= GetNSeg(); i++)
-    {
-      const Segment & seg = LineSegment (i);
+	if (surfnr == 0 || el.GetIndex() == surfnr)
+	  {
+	    for (j = 1; j <= el.GetNP(); j++)
+	      {
+		INDEX_2 seg (el.PNumMod(j), el.PNumMod(j+1));
+		INDEX_2 data;
 
-      if (surfnr == 0 || seg.si == surfnr)
-	{
-	  INDEX_2 key(seg.p2, seg.p1);
-	  if (!faceht.Used(key))
-	    {
-	      cerr << "ERROR: Segment " << seg << " brother not used" << endl;
-	      (*testout) << "ERROR: Segment " << seg << " brother not used" << endl;
-	    }
-	}
-    }
+		if (seg.I1() <= 0 || seg.I2() <= 0)
+		  cerr << "seg = " << seg << endl;
 
+		if (faceht.Used(seg))
+		  {
+		    data = faceht.Get(seg);
+		    if (data.I1() == el.GetIndex())
+		      {
+			data.I1() = 0;
+			faceht.Set (seg, data);
+		      }
+		    else
+		      {
+			PrintSysError ("hash table si not fitting for segment: ",
+				       seg.I1(), "-", seg.I2(), " other = ",
+				       data.I2());
+		      }
+		  }
+		else
+		  {
+		    Swap (seg.I1(), seg.I2());
+		    data.I1() = el.GetIndex();
+		    data.I2() = i;
 
+		    faceht.Set (seg, data);
+		  }
+	      }
+	  }
+      }  
 
-  
-  for (i = 1; i <= GetNSE(); i++)
-    {
-      const Element2d & el = SurfaceElement(i);
-      
-      if (surfnr == 0 || el.GetIndex() == surfnr)
+    (*testout) << "open segments: " << endl;
+    opensegments.SetSize(0);
+    for (i = 1; i <= faceht.GetNBags(); i++)
+      for (j = 1; j <= faceht.GetBagSize(i); j++)
 	{
-	  for (j = 1; j <= el.GetNP(); j++)
+	  INDEX_2 i2;
+	  INDEX_2 data;
+	  faceht.GetData (i, j, i2, data);
+	  if (data.I1())  // surfnr
 	    {
-	      INDEX_2 seg (el.PNumMod(j), el.PNumMod(j+1));
-	      INDEX_2 data;
+	      Segment seg;
+	      seg.p1 = i2.I1();
+	      seg.p2 = i2.I2();
+	      seg.si = data.I1();
 
-	      if (faceht.Used(seg))
+	      // find geomdata:
+	      if (data.I2() > 0)
 		{
-		  data = faceht.Get(seg);
-		  if (data.I1() == el.GetIndex())
+		  // segment due to triangle
+		  const Element2d & el = SurfaceElement (data.I2());
+		  for (k = 1; k <= el.GetNP(); k++)
 		    {
-		      data.I1() = 0;
-		      faceht.Set (seg, data);
-		    }
-		  else
-		    {
-		      PrintSysError ("hash table si not fitting for segment: ",
-				     seg.I1(), "-", seg.I2(), " other = ",
-				     data.I2());
+		      if (seg.p1 == el.PNum(k))
+			seg.geominfo[0] = el.GeomInfoPi(k);
+		      if (seg.p2 == el.PNum(k))
+			seg.geominfo[1] = el.GeomInfoPi(k);
 		    }
+
+		  (*testout) << "trig seg: ";
 		}
 	      else
 		{
-		  Swap (seg.I1(), seg.I2());
-		  data.I1() = el.GetIndex();
-		  data.I2() = i;
+		  // segment due to line
+		  const Segment & lseg = LineSegment (-data.I2());
+		  seg.geominfo[0] = lseg.geominfo[0];
+		  seg.geominfo[1] = lseg.geominfo[1];
 
-		  faceht.Set (seg, data);
+		  (*testout) << "line seg: ";
 		}
-	    }
-	}
-    }  
-  
-  (*testout) << "open segments: " << endl;
-  opensegments.SetSize(0);
-  for (i = 1; i <= faceht.GetNBags(); i++)
-    for (j = 1; j <= faceht.GetBagSize(i); j++)
-      {
-	INDEX_2 i2;
-	INDEX_2 data;
-	faceht.GetData (i, j, i2, data);
-	if (data.I1())  // surfnr
-	  {
-	    Segment seg;
-	    seg.p1 = i2.I1();
-	    seg.p2 = i2.I2();
-	    seg.si = data.I1();
-
-	    // find geomdata:
-	    if (data.I2() > 0)
-	      {
-		// segment due to triangle
-		const Element2d & el = SurfaceElement (data.I2());
-		for (k = 1; k <= el.GetNP(); k++)
-		  {
-		    if (seg.p1 == el.PNum(k))
-		      seg.geominfo[0] = el.GeomInfoPi(k);
-		    if (seg.p2 == el.PNum(k))
-		      seg.geominfo[1] = el.GeomInfoPi(k);
-		  }
-
-		(*testout) << "trig seg: ";
-	      }
-	    else
-	      {
-		// segment due to line
-		const Segment & lseg = LineSegment (-data.I2());
-		seg.geominfo[0] = lseg.geominfo[0];
-		seg.geominfo[1] = lseg.geominfo[1];
-
-		(*testout) << "line seg: ";
-	      }
 	    
-	    (*testout) << seg.p1 << " - " << seg.p2 
-		       << " len = " << Dist (Point(seg.p1), Point(seg.p2))
-		       << endl;
+	      (*testout) << seg.p1 << " - " << seg.p2 
+			 << " len = " << Dist (Point(seg.p1), Point(seg.p2))
+			 << endl;
 
-	    opensegments.Append (seg);
-	    if (seg.geominfo[0].trignum <= 0 || seg.geominfo[1].trignum <= 0)
-	      {
-		(*testout) << "Problem with open segment: " << seg << endl;
-	      }
+	      opensegments.Append (seg);
+	      if (seg.geominfo[0].trignum <= 0 || seg.geominfo[1].trignum <= 0)
+		{
+		  (*testout) << "Problem with open segment: " << seg << endl;
+		}
 
-	  }
-      }
+	    }
+	}
 
-  PrintMessage (3, opensegments.Size(), " open segments found");
-  (*testout) << opensegments.Size() << " open segments found" << endl;
+    PrintMessage (3, opensegments.Size(), " open segments found");
+    (*testout) << opensegments.Size() << " open segments found" << endl;
   
-  ptyps.SetSize (GetNP());
-  for (i = 1; i <= ptyps.Size(); i++)
-    ptyps.Elem(i) = SURFACEPOINT;
+    /*
+      ptyps.SetSize (GetNP());
+      for (i = 1; i <= ptyps.Size(); i++)
+      ptyps.Elem(i) = SURFACEPOINT;
 
-  for (i = 1; i <= GetNSeg(); i++)
-    {
+      for (i = 1; i <= GetNSeg(); i++)
+      {
       const Segment & seg = LineSegment (i);
       ptyps.Elem(seg.p1) = EDGEPOINT;
       ptyps.Elem(seg.p2) = EDGEPOINT;
-    }
-  for (i = 1; i <= GetNOpenSegments(); i++)
-    {
+      }
+      for (i = 1; i <= GetNOpenSegments(); i++)
+      {
       const Segment & seg = GetOpenSegment (i);
       ptyps.Elem(seg.p1) = EDGEPOINT;
       ptyps.Elem(seg.p2) = EDGEPOINT;
-    }
+      }
+    */
+    for (i = 1; i <= points.Size(); i++)
+      points.Elem(i).SetType(SURFACEPOINT);
+
+    for (i = 1; i <= GetNSeg(); i++)
+      {
+	const Segment & seg = LineSegment (i);
+	points[seg.p1].SetType(EDGEPOINT);
+	points[seg.p2].SetType(EDGEPOINT);
+      }
+    for (i = 1; i <= GetNOpenSegments(); i++)
+      {
+	const Segment & seg = GetOpenSegment (i);
+	points[seg.p1].SetType (EDGEPOINT);
+	points[seg.p2].SetType (EDGEPOINT);
+      }
   
   
-  /*
+  
+    /*
 
-  for (i = 1; i <= openelements.Size(); i++)
+    for (i = 1; i <= openelements.Size(); i++)
     {
-      const Element2d & sel = openelements.Get(i);
+    const Element2d & sel = openelements.Get(i);
 
-      if (boundaryedges)
-	for (j = 1; j <= sel.GetNP(); j++)
-	  {
-	    INDEX_2 i2;
-	    i2.I1() = sel.PNumMod(j);
-	    i2.I2() = sel.PNumMod(j+1);
-	    i2.Sort();
-	    boundaryedges->Set (i2, 1);
-	  }
+    if (boundaryedges)
+    for (j = 1; j <= sel.GetNP(); j++)
+    {
+    INDEX_2 i2;
+    i2.I1() = sel.PNumMod(j);
+    i2.I2() = sel.PNumMod(j+1);
+    i2.Sort();
+    boundaryedges->Set (i2, 1);
+    }
       
-      for (j = 1; j <= 3; j++)
-	{
-	  int pi = sel.PNum(j);
-	  if (pi <= ptyps.Size())
-	    ptyps.Elem(pi) = FIXEDPOINT;
-	}
+    for (j = 1; j <= 3; j++)
+    {
+    int pi = sel.PNum(j);
+    if (pi <= ptyps.Size())
+    ptyps.Elem(pi) = FIXEDPOINT;
     }
-  */
-}
+    }
+    */
+  }
 
 
-void Mesh :: RemoveOneLayerSurfaceElements ()
-{
-  int i, j;
-  int np = GetNP();
+  void Mesh :: RemoveOneLayerSurfaceElements ()
+  {
+    int i, j;
+    int np = GetNP();
 
-  FindOpenSegments();
-  BitArray frontpoints(np);
+    FindOpenSegments();
+    BitArray frontpoints(np);
 
-  frontpoints.Clear();
-  for (i = 1; i <= GetNOpenSegments(); i++)
-    {
-      const Segment & seg = GetOpenSegment(i);
-      frontpoints.Set (seg.p1);
-      frontpoints.Set (seg.p2);
-    }
+    frontpoints.Clear();
+    for (i = 1; i <= GetNOpenSegments(); i++)
+      {
+	const Segment & seg = GetOpenSegment(i);
+	frontpoints.Set (seg.p1);
+	frontpoints.Set (seg.p2);
+      }
 
-  for (i = 1; i <= GetNSE(); i++)
-    {
-      Element2d & sel = surfelements.Elem(i);
-      int remove = 0;
-      for (j = 1; j <= sel.GetNP(); j++)
-	if (frontpoints.Test(sel.PNum(j)))
-	  remove = 1;
-      if (remove)
-	sel.PNum(1) = 0;
-    }
+    for (i = 1; i <= GetNSE(); i++)
+      {
+	Element2d & sel = surfelements.Elem(i);
+	int remove = 0;
+	for (j = 1; j <= sel.GetNP(); j++)
+	  if (frontpoints.Test(sel.PNum(j)))
+	    remove = 1;
+	if (remove)
+	  sel.PNum(1) = 0;
+      }
 
-  for (i = surfelements.Size(); i >= 1; i--)
-    {
-      if (surfelements.Elem(i).PNum(1) == 0)
-	{
-	  surfelements.Elem(i) = surfelements.Last();
-	  surfelements.DeleteLast();
-	}
-    }
-  timestamp = NextTimeStamp();
-  //  Compress();
-}
+    for (i = surfelements.Size(); i >= 1; i--)
+      {
+	if (surfelements.Elem(i).PNum(1) == 0)
+	  {
+	    surfelements.Elem(i) = surfelements.Last();
+	    surfelements.DeleteLast();
+	  }
+      }
+    timestamp = NextTimeStamp();
+    //  Compress();
+  }
 
 
 
 
 
-void Mesh :: FreeOpenElementsEnvironment (int layers)
-{
-  int i, j, k;
-  PointIndex pi;
-  const int large = 9999;
-  ARRAY<int,PointIndex::BASE> dist(GetNP());
+  void Mesh :: FreeOpenElementsEnvironment (int layers)
+  {
+    int i, j, k;
+    PointIndex pi;
+    const int large = 9999;
+    ARRAY<int,PointIndex::BASE> dist(GetNP());
 
-  dist = large;
+    dist = large;
 
-  for (i = 1; i <= GetNOpenElements(); i++)
-    {
-      const Element2d & face = OpenElement(i);
-      for (j = 1; j <= face.GetNP(); j++)
-	dist[face.PNum(j)] = 1;
-    }
+    for (i = 1; i <= GetNOpenElements(); i++)
+      {
+	const Element2d & face = OpenElement(i);
+	for (j = 1; j <= face.GetNP(); j++)
+	  dist[face.PNum(j)] = 1;
+      }
+
+    for (k = 1; k <= layers; k++)
+      for (i = 1; i <= GetNE(); i++)
+	{
+	  const Element & el = VolumeElement(i);
+	  if (el[0] == -1 || el.IsDeleted()) continue;
+
+	  int elmin = large;
+	  for (j = 0; j < el.GetNP(); j++)
+	    if (dist[el[j]] < elmin)
+	      elmin = dist[el[j]];
+
+	  if (elmin < large)
+	    {
+	      for (j = 0; j < el.GetNP(); j++)
+		if (dist[el[j]] > elmin+1)
+		  dist[el[j]] = elmin+1;
+	    }
+	}
 
-  for (k = 1; k <= layers; k++)
+    int cntfree = 0;
     for (i = 1; i <= GetNE(); i++)
       {
 	const Element & el = VolumeElement(i);
 	if (el[0] == -1 || el.IsDeleted()) continue;
-
+	
 	int elmin = large;
 	for (j = 0; j < el.GetNP(); j++)
 	  if (dist[el[j]] < elmin)
 	    elmin = dist[el[j]];
-
-	if (elmin < large)
-	  {
-	    for (j = 0; j < el.GetNP(); j++)
-	      if (dist[el[j]] > elmin+1)
-		dist[el[j]] = elmin+1;
-	  }
-      }
-
-  int cntfree = 0;
-  for (i = 1; i <= GetNE(); i++)
-    {
-      const Element & el = VolumeElement(i);
-      if (el[0] == -1 || el.IsDeleted()) continue;
-	
-      int elmin = large;
-      for (j = 0; j < el.GetNP(); j++)
-	if (dist[el[j]] < elmin)
-	  elmin = dist[el[j]];
       
-      eltyps.Elem(i) = (elmin <= layers) ? 
-	FREEELEMENT : FIXEDELEMENT;
-      if (elmin <= layers)
-	cntfree++;
-    }
+	eltyps.Elem(i) = (elmin <= layers) ? 
+	  FREEELEMENT : FIXEDELEMENT;
+	if (elmin <= layers)
+	  cntfree++;
+      }
 
-  PrintMessage (5, "free: ", cntfree, ", fixed: ", GetNE()-cntfree);
-  (*testout) << "free: " << cntfree << ", fixed: " << GetNE()-cntfree << endl;
+    PrintMessage (5, "free: ", cntfree, ", fixed: ", GetNE()-cntfree);
+    (*testout) << "free: " << cntfree << ", fixed: " << GetNE()-cntfree << endl;
 
-  for (pi = PointIndex::BASE; 
-       pi < GetNP()+PointIndex::BASE; pi++)
-    {
-      if (dist[pi] > layers+1)
-	ptyps[pi] = FIXEDPOINT;
-    }
-}
+    for (pi = PointIndex::BASE; 
+	 pi < GetNP()+PointIndex::BASE; pi++)
+      {
+	if (dist[pi] > layers+1)
+	  points[pi].SetType(FIXEDPOINT);
+      }
+  }
 
 
 
-void Mesh :: SetLocalH (const Point3d & pmin, const Point3d & pmax, double grading)
-{
-  Point3d c = Center (pmin, pmax);
-  double d = max3 (pmax.X()-pmin.X(),
-		   pmax.Y()-pmin.Y(),
-		   pmax.Z()-pmin.Z());
-  d /= 2;
-  Point3d pmin2 = c - Vec3d (d, d, d);
-  Point3d pmax2 = c + Vec3d (d, d, d);
+  void Mesh :: SetLocalH (const Point3d & pmin, const Point3d & pmax, double grading)
+  {
+    Point3d c = Center (pmin, pmax);
+    double d = max3 (pmax.X()-pmin.X(),
+		     pmax.Y()-pmin.Y(),
+		     pmax.Z()-pmin.Z());
+    d /= 2;
+    Point3d pmin2 = c - Vec3d (d, d, d);
+    Point3d pmax2 = c + Vec3d (d, d, d);
   
 
-  delete lochfunc;
-  lochfunc = new LocalH (pmin2, pmax2, grading);
-}
+    delete lochfunc;
+    lochfunc = new LocalH (pmin2, pmax2, grading);
+  }
 
-void Mesh :: RestrictLocalH (const Point3d & p, double hloc)
-{
-  // cout << "restrict h in " << p << " to " << hloc << endl;
-  if (!lochfunc)
-    {
-      PrintWarning("RestrictLocalH called, creating mesh-size tree");
+  void Mesh :: RestrictLocalH (const Point3d & p, double hloc)
+  {
+    //cout << "restrict h in " << p << " to " << hloc << endl;
+    if (!lochfunc)
+      {
+	PrintWarning("RestrictLocalH called, creating mesh-size tree");
 
-      Point3d boxmin, boxmax;
-      GetBox (boxmin, boxmax);
-      SetLocalH (boxmin, boxmax, 0.8);
-    }
+	Point3d boxmin, boxmax;
+	GetBox (boxmin, boxmax);
+	SetLocalH (boxmin, boxmax, 0.8);
+      }
 
-  lochfunc -> SetH (p, hloc);
-}
+    lochfunc -> SetH (p, hloc);
+  }
 
-void Mesh :: RestrictLocalHLine (const Point3d & p1, 
-				 const Point3d & p2,
-				 double hloc)
-{
-  // cout << "restrict h along " << p1 << " - " << p2 << " to " << hloc << endl;
-  int i;
-  int steps = int (Dist (p1, p2) / hloc) + 2;
-  Vec3d v(p1, p2);
+  void Mesh :: RestrictLocalHLine (const Point3d & p1, 
+				   const Point3d & p2,
+				   double hloc)
+  {
+    // cout << "restrict h along " << p1 << " - " << p2 << " to " << hloc << endl;
+    int i;
+    int steps = int (Dist (p1, p2) / hloc) + 2;
+    Vec3d v(p1, p2);
   
-  for (i = 0; i <= steps; i++)
-    {
-      Point3d p = p1 + (double(i)/double(steps) * v);
-      RestrictLocalH (p, hloc);
-    }
-}
-
-
-void Mesh :: SetGlobalH (double h)
-{
-  hglob = h;
-}
+    for (i = 0; i <= steps; i++)
+      {
+	Point3d p = p1 + (double(i)/double(steps) * v);
+	RestrictLocalH (p, hloc);
+      }
+  }
 
-double Mesh :: MaxHDomain (int dom) const
-{
-  if (maxhdomain.Size())
-    return maxhdomain.Get(dom);
-  else
-    return 1e10;
-}
 
-void Mesh :: SetMaxHDomain (const ARRAY<double> & mhd)
-{
-  maxhdomain.SetSize(mhd.Size());
-  for (int i = 1; i <= mhd.Size(); i++)
-    maxhdomain.Elem(i) = mhd.Get(i);
-}
+  void Mesh :: SetGlobalH (double h)
+  {
+    hglob = h;
+  }
 
+  double Mesh :: MaxHDomain (int dom) const
+  {
+    if (maxhdomain.Size())
+      return maxhdomain.Get(dom);
+    else
+      return 1e10;
+  }
 
-double Mesh :: GetH (const Point3d & p) const
-{
-  double hmin = hglob;
-  if (lochfunc)
-    {
-      double hl = lochfunc->GetH (p);
-      if (hl < hglob)
-	hmin = hl;
-    }
-  return hmin;
-}
+  void Mesh :: SetMaxHDomain (const ARRAY<double> & mhd)
+  {
+    maxhdomain.SetSize(mhd.Size());
+    for (int i = 1; i <= mhd.Size(); i++)
+      maxhdomain.Elem(i) = mhd.Get(i);
+  }
 
-double Mesh :: GetMinH (const Point3d & pmin, const Point3d & pmax)
-{
-  double hmin = hglob;
-  if (lochfunc)
-    {
-      double hl = lochfunc->GetMinH (pmin, pmax);
-      if (hl < hmin)
-	hmin = hl;
-    }
-  return hmin;
-}
 
+  double Mesh :: GetH (const Point3d & p) const
+  {
+    double hmin = hglob;
+    if (lochfunc)
+      {
+	double hl = lochfunc->GetH (p);
+	if (hl < hglob)
+	  hmin = hl;
+      }
+    return hmin;
+  }
 
+  double Mesh :: GetMinH (const Point3d & pmin, const Point3d & pmax)
+  {
+    double hmin = hglob;
+    if (lochfunc)
+      {
+	double hl = lochfunc->GetMinH (pmin, pmax);
+	if (hl < hmin)
+	  hmin = hl;
+      }
+    return hmin;
+  }
 
 
 
-double Mesh :: AverageH (int surfnr) const
-{
-  int i, j, n;
-  double hi, hsum;
-  double maxh = 0, minh = 1e10;
 
-  hsum = 0;
-  n = 0;
-  for (i = 1; i <= GetNSE(); i++)
-    {
-      const Element2d & el = SurfaceElement(i);
-      if (surfnr == 0 || el.GetIndex() == surfnr)
-	{
-	  for (j = 1; j <= 3; j++)
-	    {
-	      hi = Dist (Point (el.PNumMod(j)), 
-			 Point (el.PNumMod(j+1)));
 
-	      hsum += hi;
+  double Mesh :: AverageH (int surfnr) const
+  {
+    int i, j, n;
+    double hi, hsum;
+    double maxh = 0, minh = 1e10;
 
-	      if (hi > maxh) maxh = hi;
-	      if (hi < minh) minh = hi;
-	      n++;
-	    }
-	}
-    }
+    hsum = 0;
+    n = 0;
+    for (i = 1; i <= GetNSE(); i++)
+      {
+	const Element2d & el = SurfaceElement(i);
+	if (surfnr == 0 || el.GetIndex() == surfnr)
+	  {
+	    for (j = 1; j <= 3; j++)
+	      {
+		hi = Dist (Point (el.PNumMod(j)), 
+			   Point (el.PNumMod(j+1)));
 
-  PrintMessage (5, "minh = ", minh, " avh = ", (hsum/n), " maxh = ", maxh);
-  return (hsum / n);
-}
+		hsum += hi;
 
+		if (hi > maxh) maxh = hi;
+		if (hi < minh) minh = hi;
+		n++;
+	      }
+	  }
+      }
 
+    PrintMessage (5, "minh = ", minh, " avh = ", (hsum/n), " maxh = ", maxh);
+    return (hsum / n);
+  }
 
-void Mesh :: CalcLocalH () 
-{
-  if (!lochfunc)
-    {
-      Point3d pmin, pmax;
-      GetBox (pmin, pmax);
-      SetLocalH (pmin, pmax, mparam.grading);
-    }
 
-  PrintMessage (3,
-		"CalcLocalH: ", 
-		GetNP(), " Points ", 
-		GetNE(), " Elements ", 
-		GetNSE(), " Surface Elements");
 
+  void Mesh :: CalcLocalH () 
+  {
+    if (!lochfunc)
+      {
+	Point3d pmin, pmax;
+	GetBox (pmin, pmax);
+	SetLocalH (pmin, pmax, mparam.grading);
+      }
 
-  int i;
-  for (i = 0; i < GetNSE(); i++)
-    {
-      const Element2d & el = surfelements[i];
-      int j;
+    PrintMessage (3,
+		  "CalcLocalH: ", 
+		  GetNP(), " Points ", 
+		  GetNE(), " Elements ", 
+		  GetNSE(), " Surface Elements");
 
-      if (el.GetNP() == 3)
-	{
-	  double hel = -1;
-	  for (j = 1; j <= 3; j++)
-	    {
-	      const Point3d & p1 = points[el.PNumMod(j)];
-	      const Point3d & p2 = points[el.PNumMod(j+1)];
+
+    int i;
+    for (i = 0; i < GetNSE(); i++)
+      {
+	const Element2d & el = surfelements[i];
+	int j;
+
+	if (el.GetNP() == 3)
+	  {
+	    double hel = -1;
+	    for (j = 1; j <= 3; j++)
+	      {
+		const Point3d & p1 = points[el.PNumMod(j)];
+		const Point3d & p2 = points[el.PNumMod(j+1)];
 	      
-	      /*
-	      INDEX_2 i21(el.PNumMod(j), el.PNumMod(j+1));
-	      INDEX_2 i22(el.PNumMod(j+1), el.PNumMod(j));
-	      if (! identifiedpoints->Used (i21) &&
+		/*
+		  INDEX_2 i21(el.PNumMod(j), el.PNumMod(j+1));
+		  INDEX_2 i22(el.PNumMod(j+1), el.PNumMod(j));
+		  if (! identifiedpoints->Used (i21) &&
 		  ! identifiedpoints->Used (i22) )
-	      */
-	      if (!ident.UsedSymmetric (el.PNumMod(j),
-					el.PNumMod(j+1)))
-		{
-		  double hedge = Dist (p1, p2);
-		  if (hedge > hel)
-		    hel = hedge;
-		  //		  lochfunc->SetH (Center (p1, p2), 2 * Dist (p1, p2));
-		  //		  (*testout) << "trigseth, p1,2 = " << el.PNumMod(j) << ", " << el.PNumMod(j+1) 
-		  //			     << " h = " << (2 * Dist(p1, p2)) << endl;
-		}
-	    }
+		*/
+		if (!ident -> UsedSymmetric (el.PNumMod(j),
+					     el.PNumMod(j+1)))
+		  {
+		    double hedge = Dist (p1, p2);
+		    if (hedge > hel)
+		      hel = hedge;
+		    //		  lochfunc->SetH (Center (p1, p2), 2 * Dist (p1, p2));
+		    //		  (*testout) << "trigseth, p1,2 = " << el.PNumMod(j) << ", " << el.PNumMod(j+1) 
+		    //			     << " h = " << (2 * Dist(p1, p2)) << endl;
+		  }
+	      }
 	  
-	  if (hel > 0)
-	    {
-	      const Point3d & p1 = points[el.PNum(1)];
-	      const Point3d & p2 = points[el.PNum(2)];
-	      const Point3d & p3 = points[el.PNum(3)];
-	      lochfunc->SetH (Center (p1, p2, p3), hel);
-	    }
-	}
-      else
+	    if (hel > 0)
+	      {
+		const Point3d & p1 = points[el.PNum(1)];
+		const Point3d & p2 = points[el.PNum(2)];
+		const Point3d & p3 = points[el.PNum(3)];
+		lochfunc->SetH (Center (p1, p2, p3), hel);
+	      }
+	  }
+	else
 	  {
 	    {
 	      const Point3d & p1 = points[el.PNum(1)];
@@ -1848,565 +2160,614 @@ void Mesh :: CalcLocalH ()
 	      lochfunc->SetH (Center (p1, p2), 2 * Dist (p1, p2));
 	    }
 	  }
-    }
+      }
 
-  for (i = 0; i < GetNSeg(); i++)
-    {
-      const Segment & seg = segments[i];
-      const Point3d & p1 = points[seg.p1];
-      const Point3d & p2 = points[seg.p2];
-      /*
-      INDEX_2 i21(seg.p1, seg.p2);
-      INDEX_2 i22(seg.p2, seg.p1);
-      if (identifiedpoints)
-	if (!identifiedpoints->Used (i21) && !identifiedpoints->Used (i22))
-      */
-      if (!ident.UsedSymmetric (seg.p1, seg.p2))
-	{
-	  lochfunc->SetH (Center (p1, p2), Dist (p1, p2));
-	}
-    }
-  /*
-  cerr << "do vol" << endl;
-  for (i = 1; i <= GetNE(); i++)
-    {
+    for (i = 0; i < GetNSeg(); i++)
+      {
+	const Segment & seg = segments[i];
+	const Point3d & p1 = points[seg.p1];
+	const Point3d & p2 = points[seg.p2];
+	/*
+	  INDEX_2 i21(seg.p1, seg.p2);
+	  INDEX_2 i22(seg.p2, seg.p1);
+	  if (identifiedpoints)
+	  if (!identifiedpoints->Used (i21) && !identifiedpoints->Used (i22))
+	*/
+	if (!ident -> UsedSymmetric (seg.p1, seg.p2))
+	  {
+	    lochfunc->SetH (Center (p1, p2), Dist (p1, p2));
+	  }
+      }
+    /*
+      cerr << "do vol" << endl;
+      for (i = 1; i <= GetNE(); i++)
+      {
       const Element & el = VolumeElement(i);
       if (el.GetType() == TET)
-	{
-	  int j, k;
-	  for (j = 2; j <= 4; j++)
-	    for (k = 1; k < j; k++)  
-	      {
-		const Point3d & p1 = Point (el.PNum(j));
-		const Point3d & p2 = Point (el.PNum(k));
-		lochfunc->SetH (Center (p1, p2), 2 * Dist (p1, p2));
-		(*testout) << "set vol h to " << (2 * Dist (p1, p2)) << endl;
+      {
+      int j, k;
+      for (j = 2; j <= 4; j++)
+      for (k = 1; k < j; k++)  
+      {
+      const Point3d & p1 = Point (el.PNum(j));
+      const Point3d & p2 = Point (el.PNum(k));
+      lochfunc->SetH (Center (p1, p2), 2 * Dist (p1, p2));
+      (*testout) << "set vol h to " << (2 * Dist (p1, p2)) << endl;
 
-	      }
-	}
-    }
-  */
+      }
+      }
+      }
+    */
 
-  /*
-  const char * meshsizefilename = 
-    globflags.GetStringFlag ("meshsize", NULL);
-  if (meshsizefilename)
-    {
+    /*
+      const char * meshsizefilename = 
+      globflags.GetStringFlag ("meshsize", NULL);
+      if (meshsizefilename)
+      {
       ifstream msf(meshsizefilename);
       if (msf)
-	{
-	  int nmsp;
-	  msf >> nmsp;
-	  for (i = 1; i <= nmsp; i++)
-	    {
-	      Point3d pi;
-	      double hi;
-	      msf >> pi.X() >> pi.Y() >> pi.Z();
-	      msf >> hi;
-	      lochfunc->SetH (pi, hi);
-	    }
-	}
-    }
-  */
-  //  lochfunc -> Convexify();
-  //  lochfunc -> PrintMemInfo (cout);
-}
+      {
+      int nmsp;
+      msf >> nmsp;
+      for (i = 1; i <= nmsp; i++)
+      {
+      Point3d pi;
+      double hi;
+      msf >> pi.X() >> pi.Y() >> pi.Z();
+      msf >> hi;
+      lochfunc->SetH (pi, hi);
+      }
+      }
+      }
+    */
+    //  lochfunc -> Convexify();
+    //  lochfunc -> PrintMemInfo (cout);
+  }
 
 
+  void Mesh :: CalcLocalHFromPointDistances(void)
+  {
+    PrintMessage (3, "Calculating local h from point distances");
+  
+    if (!lochfunc)
+      {
+	Point3d pmin, pmax;
+	GetBox (pmin, pmax);
+      
+	SetLocalH (pmin, pmax, mparam.grading);
+      }
 
-void Mesh :: CalcLocalHFromSurfaceCurvature (double elperr) 
-{
-  PrintMessage (3, "Calculating local h from Surface curvature");
+    PointIndex i,j;
+    double hl;
 
-  if (!lochfunc)
-    {
-      Point3d pmin, pmax;
-      GetBox (pmin, pmax);
+  
+    for (i = PointIndex::BASE; 
+	 i < GetNP()+PointIndex::BASE; i++)
+      {
+	for(j=i+1; j<GetNP()+PointIndex::BASE; j++)
+	  {
+	    const Point3d & p1 = points[i];
+	    const Point3d & p2 = points[j];
+	    hl = Dist(p1,p2);
+	    RestrictLocalH(p1,hl);
+	    RestrictLocalH(p2,hl);
+	    //cout << "restricted h at " << p1 << " and " << p2 << " to " << hl << endl;
+	  }
+      }
+  
+
+  }
+
+
+  void Mesh :: CalcLocalHFromSurfaceCurvature (double elperr) 
+  {
+    PrintMessage (3, "Calculating local h from surface curvature");
+
+    if (!lochfunc)
+      {
+	Point3d pmin, pmax;
+	GetBox (pmin, pmax);
       
-      SetLocalH (pmin, pmax, mparam.grading);
-    }
+	SetLocalH (pmin, pmax, mparam.grading);
+      }
 
   
-  INDEX_2_HASHTABLE<int> edges(3 * GetNP() + 2);
-  INDEX_2_HASHTABLE<int> bedges(GetNSeg() + 2);
-  int i, j;
+    INDEX_2_HASHTABLE<int> edges(3 * GetNP() + 2);
+    INDEX_2_HASHTABLE<int> bedges(GetNSeg() + 2);
+    int i, j;
 
-  for (i = 1; i <= GetNSeg(); i++)
-    {
-      const Segment & seg = LineSegment(i);
-      INDEX_2 i2(seg.p1, seg.p2);
-      i2.Sort();
-      bedges.Set (i2, 1);
-    }
-  for (i = 1; i <= GetNSE(); i++)
-    {
-      const Element2d & sel = SurfaceElement(i);
-      if (!sel.PNum(1))
-	continue;
-      for (j = 1; j <= 3; j++)
-	{
-	  INDEX_2 i2(sel.PNumMod(j), sel.PNumMod(j+1));
-	  i2.Sort();
-	  if (bedges.Used(i2)) continue;
+    for (i = 1; i <= GetNSeg(); i++)
+      {
+	const Segment & seg = LineSegment(i);
+	INDEX_2 i2(seg.p1, seg.p2);
+	i2.Sort();
+	bedges.Set (i2, 1);
+      }
+    for (i = 1; i <= GetNSE(); i++)
+      {
+	const Element2d & sel = SurfaceElement(i);
+	if (!sel.PNum(1))
+	  continue;
+	for (j = 1; j <= 3; j++)
+	  {
+	    INDEX_2 i2(sel.PNumMod(j), sel.PNumMod(j+1));
+	    i2.Sort();
+	    if (bedges.Used(i2)) continue;
 
-	  if (edges.Used(i2))
-	    {
-	      int other = edges.Get(i2);
-
-	      const Element2d & elother = SurfaceElement(other);
-
-	      int pi3 = 1;
-	      while ( (sel.PNum(pi3) == i2.I1()) || 
-		      (sel.PNum(pi3) == i2.I2()))
-		pi3++;
-	      pi3 = sel.PNum(pi3);
-
-	      int pi4 = 1;
-	      while ( (elother.PNum(pi4) == i2.I1()) || 
-		      (elother.PNum(pi4) == i2.I2()))
-		pi4++;
-	      pi4 = elother.PNum(pi4);
-
-	      double rad = ComputeCylinderRadius (Point (i2.I1()),
-						  Point (i2.I2()),
-						  Point (pi3), 
-						  Point (pi4));
+	    if (edges.Used(i2))
+	      {
+		int other = edges.Get(i2);
+
+		const Element2d & elother = SurfaceElement(other);
+
+		int pi3 = 1;
+		while ( (sel.PNum(pi3) == i2.I1()) || 
+			(sel.PNum(pi3) == i2.I2()))
+		  pi3++;
+		pi3 = sel.PNum(pi3);
+
+		int pi4 = 1;
+		while ( (elother.PNum(pi4) == i2.I1()) || 
+			(elother.PNum(pi4) == i2.I2()))
+		  pi4++;
+		pi4 = elother.PNum(pi4);
+
+		double rad = ComputeCylinderRadius (Point (i2.I1()),
+						    Point (i2.I2()),
+						    Point (pi3), 
+						    Point (pi4));
 	      
-	      RestrictLocalHLine (Point(i2.I1()), Point(i2.I2()), rad/elperr);
+		RestrictLocalHLine (Point(i2.I1()), Point(i2.I2()), rad/elperr);
 
 
-	      /*	      
-	      (*testout) << "pi1,2, 3, 4 = " << i2.I1() << ", " << i2.I2() << ", " << pi3 << ", " << pi4
-			 << " p1 = " << Point(i2.I1()) 
-			 << ", p2 = " << Point(i2.I2()) 
-		//			 << ", p3 = " << Point(pi3) 
-		//			 << ", p4 = " << Point(pi4) 
-			 << ", rad = " << rad << endl;
-	      */
-	    }
-	  else
-	    edges.Set (i2, i);
-	}
-    }
+		/*	      
+			     (*testout) << "pi1,2, 3, 4 = " << i2.I1() << ", " << i2.I2() << ", " << pi3 << ", " << pi4
+			     << " p1 = " << Point(i2.I1()) 
+			     << ", p2 = " << Point(i2.I2()) 
+			     //			 << ", p3 = " << Point(pi3) 
+			     //			 << ", p4 = " << Point(pi4) 
+			     << ", rad = " << rad << endl;
+		*/
+	      }
+	    else
+	      edges.Set (i2, i);
+	  }
+      }
 
 
-  // Restrict h due to line segments
+    // Restrict h due to line segments
 
-  for (i = 1; i <= GetNSeg(); i++)
-    {
-      const Segment & seg = LineSegment(i);
-      const Point3d & p1 = Point(seg.p1);
-      const Point3d & p2 = Point(seg.p2);
-      RestrictLocalH (Center (p1, p2),  Dist (p1, p2));
-    }
+    for (i = 1; i <= GetNSeg(); i++)
+      {
+	const Segment & seg = LineSegment(i);
+	const Point3d & p1 = Point(seg.p1);
+	const Point3d & p2 = Point(seg.p2);
+	RestrictLocalH (Center (p1, p2),  Dist (p1, p2));
+      }
 
 
 
-      /*
+    /*
 
 
-  int i, j;
-  int np = GetNP();
-  int nseg = GetNSeg();
-  int nse = GetNSE();
+    int i, j;
+    int np = GetNP();
+    int nseg = GetNSeg();
+    int nse = GetNSE();
   
-  ARRAY<Vec3d> normals(np);
-  BitArray linepoint(np);
+    ARRAY<Vec3d> normals(np);
+    BitArray linepoint(np);
 
-  linepoint.Clear();
-  for (i = 1; i <= nseg; i++)
+    linepoint.Clear();
+    for (i = 1; i <= nseg; i++)
     {
-      linepoint.Set (LineSegment(i).p1);
-      linepoint.Set (LineSegment(i).p2);
+    linepoint.Set (LineSegment(i).p1);
+    linepoint.Set (LineSegment(i).p2);
     }
 
-  for (i = 1; i <= np; i++)
+    for (i = 1; i <= np; i++)
     normals.Elem(i) = Vec3d(0,0,0);
 
-  for (i = 1; i <= nse; i++)
+    for (i = 1; i <= nse; i++)
     {
-      Element2d & el = SurfaceElement(i);
-      Vec3d nf = Cross (Vec3d (Point (el.PNum(1)), Point(el.PNum(2))),
-			Vec3d (Point (el.PNum(1)), Point(el.PNum(3))));
-      for (j = 1; j <= 3; j++)
-	normals.Elem(el.PNum(j)) += nf;
+    Element2d & el = SurfaceElement(i);
+    Vec3d nf = Cross (Vec3d (Point (el.PNum(1)), Point(el.PNum(2))),
+    Vec3d (Point (el.PNum(1)), Point(el.PNum(3))));
+    for (j = 1; j <= 3; j++)
+    normals.Elem(el.PNum(j)) += nf;
     }
 
-  for (i = 1; i <= np; i++)
+    for (i = 1; i <= np; i++)
     normals.Elem(i) /= (1e-12 + normals.Elem(i).Length());
 
-  for (i = 1; i <= nse; i++)
+    for (i = 1; i <= nse; i++)
     {
-      Element2d & el = SurfaceElement(i);
-      Vec3d nf = Cross (Vec3d (Point (el.PNum(1)), Point(el.PNum(2))),
-			Vec3d (Point (el.PNum(1)), Point(el.PNum(3))));
-      nf /= nf.Length();
-      Point3d c = Center (Point(el.PNum(1)),
-			  Point(el.PNum(2)),
-			  Point(el.PNum(3)));
+    Element2d & el = SurfaceElement(i);
+    Vec3d nf = Cross (Vec3d (Point (el.PNum(1)), Point(el.PNum(2))),
+    Vec3d (Point (el.PNum(1)), Point(el.PNum(3))));
+    nf /= nf.Length();
+    Point3d c = Center (Point(el.PNum(1)),
+    Point(el.PNum(2)),
+    Point(el.PNum(3)));
 			  
-      for (j = 1; j <= 3; j++)
+    for (j = 1; j <= 3; j++)
+    {
+    if (!linepoint.Test (el.PNum(j)))
+    {
+    double dist = Dist (c, Point(el.PNum(j)));
+    double dn = (nf - normals.Get(el.PNum(j))).Length();
+	  
+    RestrictLocalH (Point(el.PNum(j)), dist / (dn+1e-12) /elperr);
+    }
+    }
+    }
+    */
+  }
+
+
+  void Mesh :: RestrictLocalH (resthtype rht, int nr, double loch)
+  {
+    int i;
+    switch (rht)
+      {
+      case RESTRICTH_FACE:
 	{
-	  if (!linepoint.Test (el.PNum(j)))
+	  for (i = 1; i <= GetNSE(); i++)
 	    {
-	      double dist = Dist (c, Point(el.PNum(j)));
-	      double dn = (nf - normals.Get(el.PNum(j))).Length();
-	  
-	      RestrictLocalH (Point(el.PNum(j)), dist / (dn+1e-12) /elperr);
+	      const Element2d & sel = SurfaceElement(i);
+	      if (sel.GetIndex() == nr)
+		RestrictLocalH (RESTRICTH_SURFACEELEMENT, i, loch);
 	    }
+	  break;
 	}
-    }
-      */
-}
+      case RESTRICTH_EDGE:
+	{
+	  for (i = 1; i <= GetNSeg(); i++)
+	    {
+	      const Segment & seg = LineSegment(i);
+	      if (seg.edgenr == nr)
+		RestrictLocalH (RESTRICTH_SEGMENT, i, loch);
+	    }
+	  break;
+	}
+      case RESTRICTH_POINT:
+	{
+	  RestrictLocalH (Point (nr), loch);
+	  break;
+	}
+
+      case RESTRICTH_SURFACEELEMENT:
+	{
+	  const Element2d & sel = SurfaceElement(nr);
+	  Point3d p = Center (Point(sel.PNum(1)),
+			      Point(sel.PNum(2)),
+			      Point(sel.PNum(3)));
+	  RestrictLocalH (p, loch);
+	  break;
+	}
+      case RESTRICTH_SEGMENT:
+	{
+	  const Segment & seg = LineSegment(nr);
+	  RestrictLocalHLine (Point (seg.p1), Point(seg.p2), loch);
+	  break;
+	}
+      }
+  }
 
 
-void Mesh :: RestrictLocalH (resthtype rht, int nr, double loch)
-{
-  int i;
-  switch (rht)
-    {
-    case RESTRICTH_FACE:
+  void Mesh :: LoadLocalMeshSize (const char * meshsizefilename)
+  {
+    if (!meshsizefilename) return;
+
+    ifstream msf(meshsizefilename);
+
+    if (!msf) return;
+
+    PrintMessage (3, "Load local mesh-size");
+    int nmsp, nmsl;
+    msf >> nmsp;
+    for (int i = 0; i < nmsp; i++)
       {
-	for (i = 1; i <= GetNSE(); i++)
-	  {
-	    const Element2d & sel = SurfaceElement(i);
-	    if (sel.GetIndex() == nr)
-	      RestrictLocalH (RESTRICTH_SURFACEELEMENT, i, loch);
-	  }
-	break;
+	Point3d pi;
+	double hi;
+	msf >> pi.X() >> pi.Y() >> pi.Z();
+	msf >> hi;
+	if (!msf.good())
+	  throw NgException ("problem in mesh-size file\n");
+	RestrictLocalH (pi, hi);
       }
-    case RESTRICTH_EDGE:
+    msf >> nmsl;
+    for (int i = 0; i < nmsl; i++)
       {
-	for (i = 1; i <= GetNSeg(); i++)
-	  {
-	    const Segment & seg = LineSegment(i);
-	    if (seg.edgenr == nr)
-	      RestrictLocalH (RESTRICTH_SEGMENT, i, loch);
-	  }
-	break;
-      }
-    case RESTRICTH_POINT:
+	Point3d p1, p2;
+	double hi;
+	msf >> p1.X() >> p1.Y() >> p1.Z();
+	msf >> p2.X() >> p2.Y() >> p2.Z();
+	msf >> hi;
+	if (!msf.good())
+	  throw NgException ("problem in mesh-size file\n");
+	RestrictLocalHLine (p1, p2, hi);
+      }  
+  }
+
+
+
+  void Mesh :: GetBox (Point3d & pmin, Point3d & pmax, int dom) const
+  {
+    if (points.Size() == 0)
       {
-	RestrictLocalH (Point (nr), loch);
-	break;
+	pmin = pmax = Point3d(0,0,0);
+	return;
       }
 
-    case RESTRICTH_SURFACEELEMENT:
+    if (dom <= 0)
       {
-	const Element2d & sel = SurfaceElement(nr);
-	Point3d p = Center (Point(sel.PNum(1)),
-			    Point(sel.PNum(2)),
-			    Point(sel.PNum(3)));
-	RestrictLocalH (p, loch);
-	break;
+	pmin = Point3d (1e10, 1e10, 1e10);
+	pmax = Point3d (-1e10, -1e10, -1e10); 
+
+	for (PointIndex pi = PointIndex::BASE; 
+	     pi < GetNP()+PointIndex::BASE; pi++)
+	  {
+	    pmin.SetToMin ( (*this) [pi] );
+	    pmax.SetToMax ( (*this) [pi] );
+	  }
       }
-    case RESTRICTH_SEGMENT:
+    else
       {
-	const Segment & seg = LineSegment(nr);
-	RestrictLocalHLine (Point (seg.p1), Point(seg.p2), loch);
-	break;
-      }
-    }
-}
+	int j, nse = GetNSE();
+	SurfaceElementIndex sei;
 
+	pmin = Point3d (1e10, 1e10, 1e10);
+	pmax = Point3d (-1e10, -1e10, -1e10); 
+	for (sei = 0; sei < nse; sei++)
+	  {
+	    const Element2d & el = (*this)[sei];
+	    if (el.IsDeleted() ) continue;
 
-void Mesh :: LoadLocalMeshSize (istream & msf)
-{
-  PrintMessage (3, "Load local mesh-size");
-  int i, nmsp, nmsl;
-  msf >> nmsp;
-  for (i = 1; i <= nmsp; i++)
-    {
-      Point3d pi;
-      double hi;
-      msf >> pi.X() >> pi.Y() >> pi.Z();
-      msf >> hi;
-      RestrictLocalH (pi, hi);
-    }
-  msf >> nmsl;
-  for (i = 1; i <= nmsl; i++)
-    {
-      Point3d p1, p2;
-      double hi;
-      msf >> p1.X() >> p1.Y() >> p1.Z();
-      msf >> p2.X() >> p2.Y() >> p2.Z();
-      msf >> hi;
-      RestrictLocalHLine (p1, p2, hi);
-    }  
-}
+	    if (dom == -1 || el.GetIndex() == dom)
+	      {
+		for (j = 0; j < 3; j++)
+		  {
+		    pmin.SetToMin ( (*this) [el[j]] );
+		    pmax.SetToMax ( (*this) [el[j]] );
+		  }
+	      }
+	  }
+      }
+
+    if (pmin.X() > 0.5e10)
+      {
+	pmin = pmax = Point3d(0,0,0);
+      }
+  }
 
 
 
-void Mesh :: GetBox (Point3d & pmin, Point3d & pmax, int dom) const
-{
-  if (points.Size() == 0)
-    {
-      pmin = pmax = Point3d(0,0,0);
-      return;
-    }
 
-  if (dom <= 0)
-    {
-      pmin = Point3d (1e10, 1e10, 1e10);
-      pmax = Point3d (-1e10, -1e10, -1e10); 
+  void Mesh :: GetBox (Point3d & pmin, Point3d & pmax, POINTTYPE ptyp) const
+  {
+    if (points.Size() == 0)
+      {
+	pmin = pmax = Point3d(0,0,0);
+	return;
+      }
 
-      for (PointIndex pi = PointIndex::BASE; 
-	   pi < GetNP()+PointIndex::BASE; pi++)
+    pmin = Point3d (1e10, 1e10, 1e10);
+    pmax = Point3d (-1e10, -1e10, -1e10); 
+  
+    for (PointIndex pi = PointIndex::BASE; 
+	 pi < GetNP()+PointIndex::BASE; pi++)
+      if (points[pi].Type() <= ptyp)
 	{
 	  pmin.SetToMin ( (*this) [pi] );
 	  pmax.SetToMax ( (*this) [pi] );
 	}
-    }
-  else
-    {
-      int j, nse = GetNSE();
-      SurfaceElementIndex sei;
-
-      pmin = Point3d (1e10, 1e10, 1e10);
-      pmax = Point3d (-1e10, -1e10, -1e10); 
-      for (sei = 0; sei < nse; sei++)
-	{
-	  const Element2d & el = (*this)[sei];
-	  if (el.IsDeleted() ) continue;
-
-	  if (dom == -1 || el.GetIndex() == dom)
-	    {
-	      for (j = 0; j < 3; j++)
-		{
-		  pmin.SetToMin ( (*this) [el[j]] );
-		  pmax.SetToMax ( (*this) [el[j]] );
-		}
-	    }
-	}
-    }
-
-  if (pmin.X() > 0.5e10)
-    {
-      pmin = pmax = Point3d(0,0,0);
-    }
-}
-
-
+  }
 
 
-void Mesh :: GetBox (Point3d & pmin, Point3d & pmax, POINTTYPE ptyp) const
-{
-  if (points.Size() == 0)
-    {
-      pmin = pmax = Point3d(0,0,0);
-      return;
-    }
 
-  pmin = Point3d (1e10, 1e10, 1e10);
-  pmax = Point3d (-1e10, -1e10, -1e10); 
-  
-  for (PointIndex pi = PointIndex::BASE; 
-       pi < GetNP()+PointIndex::BASE; pi++)
-    if (ptyps[pi] <= ptyp)
-      {
-	pmin.SetToMin ( (*this) [pi] );
-	pmax.SetToMax ( (*this) [pi] );
-      }
-}
 
+  double Mesh :: ElementError (int eli) const
+  {
+    const Element & el = volelements.Get(eli);
+    return CalcTetBadness (points.Get(el[0]), points.Get(el[1]),
+			   points.Get(el[2]), points.Get(el[3]), -1);
+  }
 
+  void Mesh :: AddLockedPoint (PointIndex pi)
+  { 
+    lockedpoints.Append (pi); 
+  }
 
+  void Mesh :: ClearLockedPoints ()
+  { 
+    lockedpoints.SetSize (0); 
+  }
 
-double Mesh :: ElementError (int eli) const
-{
-  const Element & el = volelements.Get(eli);
-  return CalcTetBadness (points.Get(el[0]), points.Get(el[1]),
-			 points.Get(el[2]), points.Get(el[3]), -1);
-}
 
-void Mesh :: AddLockedPoint (PointIndex pi)
-{ 
-  lockedpoints.Append (pi); 
-}
 
-void Mesh :: ClearLockedPoints ()
-{ 
-  lockedpoints.SetSize (0); 
-}
+  void Mesh :: Compress ()
+  {
+    int i, j;
+    ARRAY<int,PointIndex::BASE> op2np(GetNP());
+    ARRAY<MeshPoint> hpoints;
+    BitArrayChar<PointIndex::BASE> pused(GetNP());
+
+    /*
+      (*testout) << "volels: " << endl;
+      for (i = 1; i <= volelements.Size(); i++)
+      {
+      for (j = 1; j <= volelements.Get(i).GetNP(); j++)
+      (*testout) << volelements.Get(i).PNum(j) << " ";
+      (*testout) << endl;
+      }
+      (*testout) << "np: " << GetNP() << endl;
+    */
 
+    for (i = 0; i < volelements.Size(); i++)
+      if (volelements[i][0] <= PointIndex::BASE-1 ||
+	  volelements[i].IsDeleted())
+	{
+	  volelements.Delete(i);
+	  i--;
+	}
 
 
-void Mesh :: Compress ()
-{
-  int i, j;
-  ARRAY<int,PointIndex::BASE> op2np(GetNP());
-  ARRAY<Point3d> hpoints;
-  BitArrayChar<PointIndex::BASE> pused(GetNP());
+    for (i = 0; i < surfelements.Size(); i++)
+      if (surfelements[i].IsDeleted())
+	{
+	  surfelements.Delete(i);
+	  i--;
+	}
 
-  /*
-  (*testout) << "volels: " << endl;
-  for (i = 1; i <= volelements.Size(); i++)
-    {
-      for (j = 1; j <= volelements.Get(i).GetNP(); j++)
-	(*testout) << volelements.Get(i).PNum(j) << " ";
-      (*testout) << endl;
-    }
-  (*testout) << "np: " << GetNP() << endl;
-  */
+    for (i = 0; i < segments.Size(); i++)
+      if (segments[i].p1 <= PointIndex::BASE-1)
+	{
+	  segments.Delete(i);
+	  i--;
+	}
 
+    pused.Clear();
+    for (i = 0; i < volelements.Size(); i++)
+      {
+	const Element & el = volelements[i];
+	for (j = 0; j < el.GetNP(); j++)
+	  pused.Set (el[j]);
+      }
 
-  for (i = 0; i < volelements.Size(); i++)
-    if (volelements[i][0] <= PointIndex::BASE-1 ||
-	volelements[i].IsDeleted())
+    for (i = 0; i < surfelements.Size(); i++)
       {
-	volelements.Delete(i);
-	i--;
+	const Element2d & el = surfelements[i];
+	for (j = 0; j < el.GetNP(); j++)
+	  pused.Set (el[j]);
       }
-  for (i = 0; i < surfelements.Size(); i++)
-    if (surfelements[i].IsDeleted())
+
+    for (i = 0; i < segments.Size(); i++)
       {
-	surfelements.Delete(i);
-	i--;
+	const Segment & seg = segments[i];
+	pused.Set (seg.p1);
+	pused.Set (seg.p2);
       }
-  for (i = 0; i < segments.Size(); i++)
-    if (segments[i].p1 == -1)
+
+    for (i = 0; i < openelements.Size(); i++)
       {
-	segments.Delete(i);
-	i--;
+	const Element2d & el = openelements[i];
+	for (j = 0; j < el.GetNP(); j++)
+	  pused.Set(el[j]);
       }
 
-  
-  pused.Clear();
-  for (i = 0; i < volelements.Size(); i++)
-    {
-      const Element & el = volelements[i];
-      for (j = 0; j < el.GetNP(); j++)
-	pused.Set (el[j]);
-    }
+    for (i = 0; i < lockedpoints.Size(); i++)
+      pused.Set (lockedpoints[i]);
 
-  for (i = 0; i < surfelements.Size(); i++)
-    {
-      const Element2d & el = surfelements[i];
-      for (j = 0; j < el.GetNP(); j++)
-	pused.Set (el[j]);
-    }
 
-  for (i = 0; i < segments.Size(); i++)
+    /*
+    // compress points doesn´t work for identified points !
+    if (identifiedpoints)
     {
-      const Segment & seg = segments[i];
-      pused.Set (seg.p1);
-      pused.Set (seg.p2);
-    }
-
-  for (i = 0; i < openelements.Size(); i++)
+    for (i = 1; i <= identifiedpoints->GetNBags(); i++)
+    if (identifiedpoints->GetBagSize(i))
     {
-      const Element2d & el = openelements[i];
-      for (j = 0; j < el.GetNP(); j++)
-	pused.Set(el[j]);
+    pused.Set ();
+    break;
     }
-
-  for (i = 0; i < lockedpoints.Size(); i++)
-    pused.Set (lockedpoints[i]);
-
-  /*
-  // compress points doesn´t work for identified points !
-  if (identifiedpoints)
-    {
-      for (i = 1; i <= identifiedpoints->GetNBags(); i++)
-	if (identifiedpoints->GetBagSize(i))
-	  {
-	    pused.Set ();
-	    break;
-	  }
     }
-  */
-  //  pused.Set();
+    */
+    //  pused.Set();
 
 
-  int npi = PointIndex::BASE-1;
+    int npi = PointIndex::BASE-1;
 
-  for (i = PointIndex::BASE; 
-       i < points.Size()+PointIndex::BASE; i++)
-    if (pused.Test(i))
-      {
-	npi++;
-	op2np[i] = npi;
-	hpoints.Append (points[i]);
-      }
-    else
-      op2np[i] = -1;
+    for (i = PointIndex::BASE; 
+	 i < points.Size()+PointIndex::BASE; i++)
+      if (pused.Test(i))
+	{
+	  npi++;
+	  op2np[i] = npi;
+	  hpoints.Append (points[i]);
+	}
+      else
+	op2np[i] = -1;
 
 
-  points.SetSize(0);
-  for (i = 0; i < hpoints.Size(); i++)
-    points.Append (hpoints[i]);
 
+    points.SetSize(0);
+    for (i = 0; i < hpoints.Size(); i++)
+      points.Append (hpoints[i]);
 
-  for (i = 1; i <= volelements.Size(); i++)
-    {
-      Element & el = VolumeElement(i);
-      for (j = 0; j < el.GetNP(); j++)
-	el[j] = op2np[el[j]];
-    }
 
-  for (i = 1; i <= surfelements.Size(); i++)
-    {
-      Element2d & el = SurfaceElement(i);
-      for (j = 0; j < el.GetNP(); j++)
-	el[j] = op2np[el[j]];
-    }
+    for (i = 1; i <= volelements.Size(); i++)
+      {
+	Element & el = VolumeElement(i);
+	for (j = 0; j < el.GetNP(); j++)
+	  el[j] = op2np[el[j]];
+      }
+
+    for (i = 1; i <= surfelements.Size(); i++)
+      {
+	Element2d & el = SurfaceElement(i);
+	for (j = 0; j < el.GetNP(); j++)
+	  el[j] = op2np[el[j]];
+      }
   
-  for (i = 0; i < segments.Size(); i++)
-    {
-      Segment & seg = segments[i];
-      seg.p1 = op2np[seg.p1];
-      seg.p2 = op2np[seg.p2];
-    }
+    for (i = 0; i < segments.Size(); i++)
+      {
+	Segment & seg = segments[i];
+	seg.p1 = op2np[seg.p1];
+	seg.p2 = op2np[seg.p2];
+      }
 
-  for (i = 1; i <= openelements.Size(); i++)
-    {
-      Element2d & el = openelements.Elem(i);
-      for (j = 0; j < el.GetNP(); j++)
-	el[j] = op2np[el[j]];
-    }  
+    for (i = 1; i <= openelements.Size(); i++)
+      {
+	Element2d & el = openelements.Elem(i);
+	for (j = 0; j < el.GetNP(); j++)
+	  el[j] = op2np[el[j]];
+      }  
 
 
-  for (i = 0; i < lockedpoints.Size(); i++)
-    lockedpoints[i] = op2np[lockedpoints[i]];
+    for (i = 0; i < lockedpoints.Size(); i++)
+      lockedpoints[i] = op2np[lockedpoints[i]];
 
 
+    CalcSurfacesOfNode();
 
-  CalcSurfacesOfNode();
-  //  FindOpenElements();
-  timestamp = NextTimeStamp();
 
-  (*testout) << "compress, done" << endl
-	     << "np = " << points.Size()
-	     << "ne = " << volelements.Size() << ", type.size = " << eltyps.Size()
-	     <<  "volelements = " << volelements << endl;
-}
+    //  FindOpenElements();
+    timestamp = NextTimeStamp();
+
+    /*
+      (*testout) << "compress, done" << endl
+      << "np = " << points.Size()
+      << "ne = " << volelements.Size() << ", type.size = " << eltyps.Size()
+      <<  "volelements = " << volelements << endl;
+    */
+  }
 
 
-int Mesh :: CheckConsistentBoundary () const
-{
-  int nf = GetNOpenElements();
-  INDEX_2_HASHTABLE<int> edges(nf+2);
-  int i, j;
-  INDEX_2 i2;
-  int err = 0;
+  int Mesh :: CheckConsistentBoundary () const
+  {
+    int nf = GetNOpenElements();
+    INDEX_2_HASHTABLE<int> edges(nf+2);
+    int i, j;
+    INDEX_2 i2;
+    int err = 0;
 
 
-  for (i = 1; i <= nf; i++)
-    {
-      const Element2d & sel = OpenElement(i);
+    for (i = 1; i <= nf; i++)
+      {
+	const Element2d & sel = OpenElement(i);
       
-      for (j = 1; j <= sel.GetNP(); j++)
-	{
-	  i2.I1() = sel.PNumMod(j);
-	  i2.I2() = sel.PNumMod(j+1);
+	for (j = 1; j <= sel.GetNP(); j++)
+	  {
+	    i2.I1() = sel.PNumMod(j);
+	    i2.I2() = sel.PNumMod(j+1);
 
-	  int sign = (i2.I2() > i2.I1()) ? 1 : -1;
-	  i2.Sort();
-	  if (!edges.Used (i2))
-	    edges.Set (i2, 0);
+	    int sign = (i2.I2() > i2.I1()) ? 1 : -1;
+	    i2.Sort();
+	    if (!edges.Used (i2))
+	      edges.Set (i2, 0);
 
-	  edges.Set (i2, edges.Get(i2) + sign);
-	  /*
+	    edges.Set (i2, edges.Get(i2) + sign);
+	    /*
 
 	    if (edges.Used(i2))
 	    {
@@ -2419,290 +2780,335 @@ int Mesh :: CheckConsistentBoundary () const
 	    }
 	    else
 	    {
-	    Swap (i2.I1(), i2.I2());
+	    swap (i2.I1(), i2.I2());
 	    edges.Set(i2, 1);
 	    cnt1++;
 	    }
 	    */
-	}
-    }
+	  }
+      }
 
 
-  /*
-    if (cnt1 != cnt2)
-    err = 2;
+    /*
+      if (cnt1 != cnt2)
+      err = 2;
     */
 
-  for (i = 1; i <= edges.GetNBags(); i++)
-    for (j = 1; j <= edges.GetBagSize(i); j++)
+    for (i = 1; i <= edges.GetNBags(); i++)
+      for (j = 1; j <= edges.GetBagSize(i); j++)
+	{
+	  int cnt = 0;
+	  edges.GetData (i, j, i2, cnt);
+	  if (cnt)
+	    {
+	      PrintError ("Edge ", i2.I1() , " - ", i2.I2(), " multiple times in surface mesh");
+	      err = 2;
+	    }
+	}
+
+    return err;
+  }
+
+
+
+  int Mesh :: CheckOverlappingBoundary () 
+  {
+    int i, j, k;
+
+    Point3d pmin, pmax;
+    GetBox (pmin, pmax);
+    Box3dTree setree(pmin, pmax);
+    ARRAY<int> inters;
+  
+    bool overlap = 0;
+    bool incons_layers = 0;
+
+
+    for (i = 1; i <= GetNSE(); i++)
+      SurfaceElement(i).badel = 0;
+
+
+    for (i = 1; i <= GetNSE(); i++)
       {
-	int cnt = 0;
-	edges.GetData (i, j, i2, cnt);
-	if (cnt)
+	const Element2d & tri = SurfaceElement(i);
+      
+	Point3d tpmin (Point(tri[0]));
+	Point3d tpmax (tpmin);
+
+	for (k = 1; k < tri.GetNP(); k++)
 	  {
-	    PrintError ("Edge ", i2.I1() , " - ", i2.I2(), " multiple times in surface mesh");
-	    err = 2;
+	    tpmin.SetToMin (Point (tri[k]));
+	    tpmax.SetToMax (Point (tri[k]));
 	  }
-      }
+	Vec3d diag(tpmin, tpmax);
 
-  return err;
-}
+	tpmax = tpmax + 0.1 * diag;
+	tpmin = tpmin - 0.1 * diag;
 
+	setree.Insert (tpmin, tpmax, i);
+      }
 
+    for (i = 1; i <= GetNSE(); i++)
+      {
+	const Element2d & tri = SurfaceElement(i);
+      
+	Point3d tpmin (Point(tri[0]));
+	Point3d tpmax (tpmin);
 
-int Mesh :: CheckOverlappingBoundary () 
-{
-  int i, j, k;
+	for (k = 1; k < tri.GetNP(); k++)
+	  {
+	    tpmin.SetToMin (Point (tri[k]));
+	    tpmax.SetToMax (Point (tri[k]));
+	  }
 
-  Point3d pmin, pmax;
-  GetBox (pmin, pmax);
-  Box3dTree setree(pmin, pmax);
-  ARRAY<int> inters;
-  int overlap = 0;
+	setree.GetIntersecting (tpmin, tpmax, inters);
 
-  for (i = 1; i <= GetNSE(); i++)
-    SurfaceElement(i).badel = 0;
+	for (j = 1; j <= inters.Size(); j++)
+	  {
+	    const Element2d & tri2 = SurfaceElement(inters.Get(j));	  
 
+	    if ( (*this)[tri[0]].GetLayer() != (*this)[tri2[0]].GetLayer())
+	      continue;
 
-  for (i = 1; i <= GetNSE(); i++)
-    {
-      const Element2d & tri = SurfaceElement(i);
-      
-      Point3d tpmin (Point(tri[0]));
-      Point3d tpmax (tpmin);
+	    if ( (*this)[tri[0]].GetLayer() != (*this)[tri[1]].GetLayer() ||
+		 (*this)[tri[0]].GetLayer() != (*this)[tri[2]].GetLayer())
+	      {
+		incons_layers = 1;
+		cout << "inconsistent layers in triangle" << endl;
+	      }
 
-      for (k = 1; k < tri.GetNP(); k++)
-	{
-	  tpmin.SetToMin (Point (tri[k]));
-	  tpmax.SetToMax (Point (tri[k]));
-	}
-      Vec3d diag(tpmin, tpmax);
 
-      tpmax = tpmax + 0.1 * diag;
-      tpmin = tpmin - 0.1 * diag;
+	    const Point3d *trip1[3], *trip2[3];	  
+	    for (k = 1; k <= 3; k++)
+	      {
+		trip1[k-1] = &Point (tri.PNum(k));
+		trip2[k-1] = &Point (tri2.PNum(k));
+	      }
 
-      setree.Insert (tpmin, tpmax, i);
-    }
+	    if (IntersectTriangleTriangle (&trip1[0], &trip2[0]))
+	      {
+		overlap = 1;
+		PrintWarning ("Intersecting elements" 
+			      ,i, " and ", inters.Get(j));
 
-  for (i = 1; i <= GetNSE(); i++)
-    {
-      const Element2d & tri = SurfaceElement(i);
-      
-      Point3d tpmin (Point(tri[0]));
-      Point3d tpmax (tpmin);
+		(*testout) << "Intersecting: " << endl;
+		(*testout) << "openelement " << i << " with open element " << inters.Get(j) << endl;
 
-      for (k = 1; k < tri.GetNP(); k++)
-	{
-	  tpmin.SetToMin (Point (tri[k]));
-	  tpmax.SetToMax (Point (tri[k]));
-	}
+		cout << "el1 = " << tri << endl;
+		cout << "el2 = " << tri2 << endl;
+		cout << "layer1 = " <<  (*this)[tri[0]].GetLayer() << endl;
+		cout << "layer2 = " <<  (*this)[tri2[0]].GetLayer() << endl;
 
-      setree.GetIntersecting (tpmin, tpmax, inters);
 
-      for (j = 1; j <= inters.Size(); j++)
-	{
-	  const Element2d & tri2 = SurfaceElement(inters.Get(j));	  
+		for (k = 1; k <= 3; k++)
+		  (*testout) << tri.PNum(k) << "  ";
+		(*testout) << endl;
+		for (k = 1; k <= 3; k++)
+		  (*testout) << tri2.PNum(k) << "  ";
+		(*testout) << endl;
 
-	  const Point3d *trip1[3], *trip2[3];	  
-	  for (k = 1; k <= 3; k++)
-	    {
-	      trip1[k-1] = &Point (tri.PNum(k));
-	      trip2[k-1] = &Point (tri2.PNum(k));
-	    }
+		for (k = 0; k <= 2; k++)
+		  (*testout) << *trip1[k] << "   ";
+		(*testout) << endl;
+		for (k = 0; k <= 2; k++)
+		  (*testout) << *trip2[k] << "   ";
+		(*testout) << endl;
 
-	  if (IntersectTriangleTriangle (&trip1[0], &trip2[0]))
-	    {
-	      overlap = 1;
-	      PrintWarning ("Intersecting elements" 
-			    ,i, " and ", inters.Get(j));
-
-	      (*testout) << "Intersecting: " << endl;
-	      (*testout) << "openelement " << i << " with open element " << inters.Get(j) << endl;
-
-	      for (k = 1; k <= 3; k++)
-		(*testout) << tri.PNum(k) << "  ";
-	      (*testout) << endl;
-	      for (k = 1; k <= 3; k++)
-		(*testout) << tri2.PNum(k) << "  ";
-	      (*testout) << endl;
-
-	      for (k = 0; k <= 2; k++)
-		(*testout) << *trip1[k] << "   ";
-	      (*testout) << endl;
-	      for (k = 0; k <= 2; k++)
-		(*testout) << *trip2[k] << "   ";
-	      (*testout) << endl;
-
-
-	      /*
-	      INDEX_3 i3(tri.PNum(1), tri.PNum(2), tri.PNum(3));
-	      i3.Sort();
-	      for (k = 1; k <= GetNSE(); k++)
-		{
+
+		/*
+		  INDEX_3 i3(tri.PNum(1), tri.PNum(2), tri.PNum(3));
+		  i3.Sort();
+		  for (k = 1; k <= GetNSE(); k++)
+		  {
 		  const Element2d & el2 = SurfaceElement(k);
 		  INDEX_3 i3b(el2.PNum(1), el2.PNum(2), el2.PNum(3));
 		  i3b.Sort();
 		  if (i3 == i3b)
-		    {
-		      SurfaceElement(k).badel = 1;
-		    }
-		}
-	      */
-	      SurfaceElement(i).badel = 1;
-	      SurfaceElement(inters.Get(j)).badel = 1;
-	    }
-	}
-    }
+		  {
+		  SurfaceElement(k).badel = 1;
+		  }
+		  }
+		*/
+		SurfaceElement(i).badel = 1;
+		SurfaceElement(inters.Get(j)).badel = 1;
+	      }
+	  }
+      }
 
-  return overlap;
-}
+    // bug 'fix'
+    if (incons_layers) overlap = 0;
+
+    return overlap;
+  }
 
 
-int Mesh :: CheckVolumeMesh () const
-{
-  PrintMessage (3, "Checking volume mesh");
+  int Mesh :: CheckVolumeMesh () const
+  {
+    PrintMessage (3, "Checking volume mesh");
   
-  int ne = GetNE();
-  DenseMatrix dtrans(3,3);
-  int i, j;
+    int ne = GetNE();
+    DenseMatrix dtrans(3,3);
+    int i, j;
 
-  PrintMessage (5, "elements: ", ne);
-  for (i = 1; i <= ne; i++)
-    {
-      Element & el = (Element&) VolumeElement(i);
-      el.flags.badel = 0;
-      int nip = el.GetNIP();
-      for (j = 1; j <= nip; j++)
-	{
-	  el.GetTransformation (j, Points(), dtrans);
-	  double det = dtrans.Det();
-	  if (det > 0)
-	    {
-	      PrintError ("Element ", i , " has wrong orientation");
-	      el.flags.badel = 1;
-	    }
-	}
-    }
+    PrintMessage (5, "elements: ", ne);
+    for (i = 1; i <= ne; i++)
+      {
+	Element & el = (Element&) VolumeElement(i);
+	el.flags.badel = 0;
+	int nip = el.GetNIP();
+	for (j = 1; j <= nip; j++)
+	  {
+	    el.GetTransformation (j, Points(), dtrans);
+	    double det = dtrans.Det();
+	    if (det > 0)
+	      {
+		PrintError ("Element ", i , " has wrong orientation");
+		el.flags.badel = 1;
+	      }
+	  }
+      }
 
-  return 0;
-}
+    return 0;
+  }
 
 
-bool Mesh :: LegalTrig (const Element2d & el) const
-{
-  return 1;
-  if ( /* hp */ 1)  // needed for old, simple hp-refinement
-    { 
-      // trigs with 2 or more segments are illegal
-      int i;
-      int nseg = 0;
-
-      if (!segmentht)
-	{
-	  cerr << "no segmentht allocated" << endl;
-	  return 0;
-	}
+  bool Mesh :: LegalTrig (const Element2d & el) const
+  {
+    return 1;
+    if ( /* hp */ 1)  // needed for old, simple hp-refinement
+      { 
+	// trigs with 2 or more segments are illegal
+	int i;
+	int nseg = 0;
+
+	if (!segmentht)
+	  {
+	    cerr << "no segmentht allocated" << endl;
+	    return 0;
+	  }
 
-      //      Point3d cp(0.5, 0.5, 0.5);
-      for (i = 1; i <= 3; i++)
-	{
-	  INDEX_2 i2(el.PNumMod (i), el.PNumMod (i+1));
-	  i2.Sort();
-	  if (segmentht -> Used (i2))
-	    nseg++;
-	}
-      if (nseg >= 2) 
-	return 0;
-    }
-  return 1;
-}
+	//      Point3d cp(0.5, 0.5, 0.5);
+	for (i = 1; i <= 3; i++)
+	  {
+	    INDEX_2 i2(el.PNumMod (i), el.PNumMod (i+1));
+	    i2.Sort();
+	    if (segmentht -> Used (i2))
+	      nseg++;
+	  }
+	if (nseg >= 2) 
+	  return 0;
+      }
+    return 1;
+  }
 
-///
-bool Mesh :: LegalTet2 (Element & el) const
-{
-  //  return 1;
-  // Test, whether 4 points have a common surface plus
-  // at least 4 edges at the boundary
+  ///
+  bool Mesh :: LegalTet2 (Element & el) const
+  {
+    //  return 1;
+    // Test, whether 4 points have a common surface plus
+    // at least 4 edges at the boundary
 
-  int i, j, k;
+    int i, j, k;
   
-  // non-tets are always legal
-  if (el.GetType() != TET)
-    {
-      el.SetLegal (1);
-      return 1;
-    }
-
-  // element has at least 2 inner points ---> legal
-  int cnti = 0;
-  for (j = 1; j <= 4; j++)
-    if (PointType(el.PNum(j)) == INNERPOINT)
-      cnti++;
-  if (cnti >= 2)
-    {
-      el.SetLegal (1);
-      return 1;
-    }
+    // non-tets are always legal
+    if (el.GetType() != TET)
+      {
+	el.SetLegal (1);
+	return 1;
+      }
 
-  // which faces are boundary faces ?
-  Element2d face;
-  int bface[4];
+    // element has at least 2 inner points ---> legal
+    int cnti = 0;
+    for (j = 0; j < 4; j++)
+      if (PointType(el[j]) == INNERPOINT)
+	cnti++;
+    if (cnti >= 2)
+      {
+	el.SetLegal (1);
+	return 1;
+      }
 
-  for (i = 1; i <= 4; i++)
-    {
-      el.GetFace (i, face);
-      INDEX_3 i3 (face.PNum(1), face.PNum(2), face.PNum(3));
-      i3.Sort();
-      bface[i-1] = surfelementht->Used (i3);
-    }
+    // which faces are boundary faces ?
+    Element2d face;
+    int bface[4];
 
-  int bedge[4][4];
-  int segedge[4][4];
-  for (i = 1; i <= 4; i++)
-    for (j = 1; j < i; j++)
+    for (i = 1; i <= 4; i++)
       {
-	INDEX_2 i2(el.PNum(i), el.PNum(j));
-	i2.Sort();
+	el.GetFace (i, face);
+	INDEX_3 i3 (face.PNum(1), face.PNum(2), face.PNum(3));
+	i3.Sort();
+	bface[i-1] = surfelementht->Used (i3);
+      }
+
+    int bedge[4][4];
+    int segedge[4][4];
+    for (i = 0; i < 4; i++)
+      for (j = 0; j < i; j++)
+	{
+	  bool sege = 0, be = 0;
 
-	int sege = 0, be = 0;
+	  INDEX_2 i2(el[i], el[j]);
+	  i2.Sort();
 
-	if (boundaryedges -> Used(i2))
-	  {
-	    int val = boundaryedges -> Get(i2);
+	  /*	
+	    if (boundaryedges -> Used(i2))
+	    {
 	    be = 1;
-	    if (val == 2)
-	      sege = 1;
-	  }
+	    if (boundaryedges -> Get(i2) == 2)
+	    sege = 1;
+	    }
+	  */
+	  int pos = boundaryedges -> Position(i2);
+	  if (pos)
+	    {
+	      be = 1;
+	      if (boundaryedges -> GetData(pos) == 2)
+		sege = 1;
+	    }
 
-	segedge[j-1][i-1] =
-	  segedge[i-1][j-1] = sege;
+	  segedge[j][i] = segedge[i][j] = sege;
+	  bedge[j][i] = bedge[i][j] = be;
+	}
 
-	bedge[j-1][i-1] =
-	  bedge[i-1][j-1] = be;
-      }
+    // two boundary faces and no edge is illegal
+    for (i = 0; i < 3; i++)
+      for (j = i+1; j < 4; j++)
+	{
+	  if (bface[i] && bface[j])
+	    {
+	      // common nodes:
+	      int pi1 = 0, pi2;
+	      while (pi1 == i || pi1 == j)
+		pi1++;
+	      pi2 = 6 - i - j - pi1;
+	      if (!segedge[pi1][pi2])
+		{
+		  // 2 boundary faces withoud edge in between
+		  el.SetLegal (0);
+		  return 0;
+		}
+	    }
+	}
 
 
-  // two boundary faces and no edge is illegal
-  for (i = 0; i < 3; i++)
-    for (j = i+1; j < 4; j++)
+    // three boundary edges meeting in a Surface point
+    for (i = 0; i < 4; i++)
       {
-	if (bface[i] && bface[j])
+	bool alledges = 1;
+	if (PointType(el[i]) == SURFACEPOINT)
 	  {
-	    // common nodes:
-	    int pi1 = 0, pi2;
-	    while (pi1 == i || pi1 == j)
-	      pi1++;
-	    pi2 = 6 - i - j - pi1;
-	    /*
-	    INDEX_2 i2(el.PNum(pi1+1), el.PNum(pi2+1));
-	    i2.Sort();
-	    if (!segmentht->Used (i2))
-	    */
-	    if (!segedge[pi1][pi2])
+	    for (j = 0; j < 4; j++)
+	      if (j != i)
+		{
+		  if (!bedge[i][j])
+		    {
+		      alledges = 0;
+		      break;
+		    }
+		}
+	    if (alledges)
 	      {
-		// 2 boundary faces withoud edge in between
-		//		cout << "p1, p2 = " << pi1 << "," << pi2 << endl;
-		//		cout << "tet illegal due to first case" << endl;
+		//	      cout << "tet illegal due to unmarked node" << endl;
 		el.SetLegal (0);
 		return 0;
 	      }
@@ -2710,1456 +3116,1629 @@ bool Mesh :: LegalTet2 (Element & el) const
       }
 
 
-  // three boundary edges meeting in a Surface point
-  for (i = 1; i <= 4; i++)
-    {
-      int alledges = 1;
-      if (PointType(el.PNum(i)) == SURFACEPOINT)
+    /*  
 	{
-	  for (j = 1; j <= 4; j++)
-	    if (j != i)
-	      {
-		/*
-		INDEX_2 i2(el.PNum(i), el.PNum(j));
-		i2.Sort();
-		if (!boundaryedges->Used(i2))
-		*/
-		if (!bedge[i-1][j-1])
-		  {
-		    alledges = 0;
-		    break;
-		  }
-	      }
-	  if (alledges)
-	    {
-	      //	      cout << "tet illegal due to unmarked node" << endl;
-	      el.SetLegal (0);
-	      return 0;
-	    }
-	}
-    }
+	// having 3 boundary edges and 4 surface nodes ???
+	int nodehasedge[4];
+	int canbe = 1;   // can be that illegal tet
 
-
-  /*  
-  {
-    // having 3 boundary edges and 4 surface nodes ???
-    int nodehasedge[4];
-    int canbe = 1;   // can be that illegal tet
-
-    for (i = 0; i < 4; i++)
-      nodehasedge[i] = 0;
-    for (i = 1; i <= 4; i++)
-      {
+	for (i = 0; i < 4; i++)
+	nodehasedge[i] = 0;
+	for (i = 1; i <= 4; i++)
+	{
 	if (PointType(el.PNum(i)) != SURFACEPOINT)
-	  canbe = 0;
+	canbe = 0;
 	for (j = i+1; j <= 4; j++)
-	  {
-	    INDEX_2 i2(el.PNum(i), el.PNum(j));
-	    i2.Sort();
-	    if (boundaryedges->Used(i2))
-	      {
-		nodehasedge[i-1] = 1;
-		nodehasedge[j-1] = 1;
-	      }
+	{
+	INDEX_2 i2(el.PNum(i), el.PNum(j));
+	i2.Sort();
+	if (boundaryedges->Used(i2))
+	{
+	nodehasedge[i-1] = 1;
+	nodehasedge[j-1] = 1;
+	}
 
-	  }
-      }
-    for (i = 0; i < 4; i++)
-      if (!nodehasedge[i])
+	}
+	}
+	for (i = 0; i < 4; i++)
+	if (!nodehasedge[i])
 	canbe = 0;
 
-    if (canbe) return 0;
+	if (canbe) return 0;
     
-  }
-  */
+	}
+    */
 
-  {
-    // two connected edges on surface, but no face
+    {
+      // two connected edges on surface, but no face
 
-    int ltestmode = 0; // (el.PNum(1) == 10516);
+      int ltestmode = 0; // (el.PNum(1) == 10516);
 
-    if (ltestmode)
-      {
-	(*testout) << "pnums: " << endl;
-	for (i = 1; i <= 4; i++)
-	  (*testout) << el.PNum(i) << " ";
-	(*testout) << endl;
-      }
+      if (ltestmode)
+	{
+	  (*testout) << "pnums: " << endl;
+	  for (i = 1; i <= 4; i++)
+	    (*testout) << el.PNum(i) << " ";
+	  (*testout) << endl;
+	}
 
-    for (i = 1; i <= 4; i++)
-      if (PointType(el.PNum(i)) == SURFACEPOINT)
-	for (j = 1; j <= 4; j++)
-	  if (j != i)
-	    for (k = j+1; k <= 4; k++)
-	      if (k != i)
-		{
-		  int fnr = 10 - i - j - k;
+      for (i = 1; i <= 4; i++)
+	if (PointType(el.PNum(i)) == SURFACEPOINT)
+	  for (j = 1; j <= 4; j++)
+	    if (j != i)
+	      for (k = j+1; k <= 4; k++)
+		if (k != i)
+		  {
+		    int fnr = 10 - i - j - k;
 
-		  if (!bface[fnr-1] &&
-		      bedge[i-1][j-1] &&
-		      bedge[i-1][k-1])
-		    {
-		      el.SetLegal (0);
-		      return 0;
-		    }
-		  /*
-		  INDEX_2 e1(el.PNum(i), el.PNum(j));
-		  e1.Sort();
-		  INDEX_2 e2(el.PNum(i), el.PNum(k));
-		  e2.Sort();
-		  INDEX_3 face(el.PNum(i), el.PNum(j), el.PNum(k));
-		  face.Sort();
+		    if (!bface[fnr-1] &&
+			bedge[i-1][j-1] &&
+			bedge[i-1][k-1])
+		      {
+			el.SetLegal (0);
+			return 0;
+		      }
+		    /*
+		      INDEX_2 e1(el.PNum(i), el.PNum(j));
+		      e1.Sort();
+		      INDEX_2 e2(el.PNum(i), el.PNum(k));
+		      e2.Sort();
+		      INDEX_3 face(el.PNum(i), el.PNum(j), el.PNum(k));
+		      face.Sort();
 		  
-		  if (ltestmode)
-		    {
+		      if (ltestmode)
+		      {
 		      (*testout) << "i, j, k = " << i << ", " << j << ", " << k << endl;
 		      (*testout) << "eij = " << boundaryedges->Used(e1) 
-				 << " eik = " << boundaryedges->Used(e2) 
-				 << " face = " << surfelementht->Used (face) << endl;
+		      << " eik = " << boundaryedges->Used(e2) 
+		      << " face = " << surfelementht->Used (face) << endl;
 		      
-		    }
+		      }
 		  
-		  if (boundaryedges->Used(e1) && 
+		      if (boundaryedges->Used(e1) && 
 		      boundaryedges->Used(e2) &&
 		      !surfelementht->Used (face))
-		    {
+		      {
 		      //		      cout << "tet illegal due to last case" << endl;
 		      el.SetLegal (0);
 		      return 0;
-		    }
-		  */
-		}
+		      }
+		    */
+		  }
     
-  }
+    }
 
 
-  {
-    // connected surface edge and edge edge, but no face
+    {
+      // connected surface edge and edge edge, but no face
 
-    for (i = 1; i <= 4; i++)
-      if (PointType(el.PNum(i)) == EDGEPOINT)
-	for (j = 1; j <= 4; j++)
-	  if (j != i)
-	    for (k = j+1; k <= 4; k++)
-	      if (k != i)
-		{
-		  int fnr = 10 - i - j - k;
+      for (i = 1; i <= 4; i++)
+	if (PointType(el.PNum(i)) == EDGEPOINT)
+	  for (j = 1; j <= 4; j++)
+	    if (j != i)
+	      for (k = j+1; k <= 4; k++)
+		if (k != i)
+		  {
+		    int fnr = 10 - i - j - k;
 
-		  if (!bface[fnr-1] &&
-		      (bedge[i-1][j-1] && segedge[i-1][k-1] ||
-		       segedge[i-1][j-1] && bedge[i-1][k-1]))
-		    {
-		      el.SetLegal (0);
-		      return 0;
-		    }
-		}
+		    if (!bface[fnr-1] &&
+			(bedge[i-1][j-1] && segedge[i-1][k-1] ||
+			 segedge[i-1][j-1] && bedge[i-1][k-1]))
+		      {
+			el.SetLegal (0);
+			return 0;
+		      }
+		  }
     
-  }
+    }
 
 
 
 
 
     
-  el.SetLegal (1);
-  return 1;
+    el.SetLegal (1);
+    return 1;
   
-      /*
-  int i1, i2, i3, i4, j;
-  if (PointType(el.PNum(1)) != INNERPOINT && 
+    /*
+      int i1, i2, i3, i4, j;
+      if (PointType(el.PNum(1)) != INNERPOINT && 
       PointType(el.PNum(2)) != INNERPOINT && 
       PointType(el.PNum(3)) != INNERPOINT &&
       PointType(el.PNum(4)) != INNERPOINT)
-    {      
+      {      
       for (i1 = 1; i1 <= surfacesonnode.EntrySize(el.PNum(1)); i1++)
-	for (i2 = 1; i2 <= surfacesonnode.EntrySize(el.PNum(2)); i2++)
-	  if (surfacesonnode.Get(el.PNum(1), i1) == 
-	      surfacesonnode.Get(el.PNum(2), i2))
-	    for (i3 = 1; i3 <= surfacesonnode.EntrySize(el.PNum(3)); i3++)
-	      if (surfacesonnode.Get(el.PNum(1), i1) == 
-		  surfacesonnode.Get(el.PNum(3), i3))
-		for (i4 = 1; i4 <= surfacesonnode.EntrySize(el.PNum(4)); i4++)
-		  if (surfacesonnode.Get(el.PNum(1), i1) == 
-		      surfacesonnode.Get(el.PNum(4), i4))
-		    {
-		      int j, numbe = 0;
-		      INDEX_2 i2;
+      for (i2 = 1; i2 <= surfacesonnode.EntrySize(el.PNum(2)); i2++)
+      if (surfacesonnode.Get(el.PNum(1), i1) == 
+      surfacesonnode.Get(el.PNum(2), i2))
+      for (i3 = 1; i3 <= surfacesonnode.EntrySize(el.PNum(3)); i3++)
+      if (surfacesonnode.Get(el.PNum(1), i1) == 
+      surfacesonnode.Get(el.PNum(3), i3))
+      for (i4 = 1; i4 <= surfacesonnode.EntrySize(el.PNum(4)); i4++)
+      if (surfacesonnode.Get(el.PNum(1), i1) == 
+      surfacesonnode.Get(el.PNum(4), i4))
+      {
+      int j, numbe = 0;
+      INDEX_2 i2;
 
-		      for (j = 1; j <= 6; j++)
-			{
-			  switch (j)
-			    {
-			    case 1:
-			      i2.I1() = el.PNum(1);
-			      i2.I2() = el.PNum(2); break;
-			    case 2: 
-			      i2.I1() = el.PNum(1); 
-			      i2.I2() = el.PNum(3); break;
-			    case 3: 
-			      i2.I1() = el.PNum(1); 
-			      i2.I2() = el.PNum(4); break;
-			    case 4:
-			      i2.I1() = el.PNum(2); 
-			      i2.I2() = el.PNum(3); break;
-			    case 5:
-			      i2.I1() = el.PNum(2);
-			      i2.I2() = el.PNum(4); break;
-			    case 6: 
-			      i2.I1() = el.PNum(3); 
-			      i2.I2() = el.PNum(4); break;
-			    }
+      for (j = 1; j <= 6; j++)
+      {
+      switch (j)
+      {
+      case 1:
+      i2.I1() = el.PNum(1);
+      i2.I2() = el.PNum(2); break;
+      case 2: 
+      i2.I1() = el.PNum(1); 
+      i2.I2() = el.PNum(3); break;
+      case 3: 
+      i2.I1() = el.PNum(1); 
+      i2.I2() = el.PNum(4); break;
+      case 4:
+      i2.I1() = el.PNum(2); 
+      i2.I2() = el.PNum(3); break;
+      case 5:
+      i2.I1() = el.PNum(2);
+      i2.I2() = el.PNum(4); break;
+      case 6: 
+      i2.I1() = el.PNum(3); 
+      i2.I2() = el.PNum(4); break;
+      }
 
-			  i2.Sort();
-			  if (boundaryedges->Used (i2)) numbe++;
-			}
+      i2.Sort();
+      if (boundaryedges->Used (i2)) numbe++;
+      }
 
-		      if (numbe >= 4)
-			{
-// 			  (*testout) 
-// 			    << "Tet illegal: " 
-// 			    << "mat = " << el.GetIndex() << " "
-// 			    << "surf = " << surfacesonnode.Get(el.PNum(1), i1) 
-// 			    << " " 
-// 			    << el.PNum(1) << " "
-// 			    << el.PNum(2) << " "
-// 			    << el.PNum(3) << " "
-// 			    << el.PNum(4) << endl;
-
-			  return 0;
-			}
-		    }
-    }
-  return 1;
-*/
-}
+      if (numbe >= 4)
+      {
+      // 			  (*testout) 
+      // 			    << "Tet illegal: " 
+      // 			    << "mat = " << el.GetIndex() << " "
+      // 			    << "surf = " << surfacesonnode.Get(el.PNum(1), i1) 
+      // 			    << " " 
+      // 			    << el.PNum(1) << " "
+      // 			    << el.PNum(2) << " "
+      // 			    << el.PNum(3) << " "
+      // 			    << el.PNum(4) << endl;
 
+      return 0;
+      }
+      }
+      }
+      return 1;
+    */
+  }
 
-int Mesh :: GetNDomains() const
-{
-  int ndom = 0;
 
-  for (int k = 0; k < facedecoding.Size(); k++)
-    {
-      if (facedecoding[k].DomainIn() > ndom)
-	ndom = facedecoding[k].DomainIn();
-      if (facedecoding[k].DomainOut() > ndom)
-	ndom = facedecoding[k].DomainOut();
-    }
+  int Mesh :: GetNDomains() const
+  {
+    int ndom = 0;
 
-  return ndom;
-}
+    for (int k = 0; k < facedecoding.Size(); k++)
+      {
+	if (facedecoding[k].DomainIn() > ndom)
+	  ndom = facedecoding[k].DomainIn();
+	if (facedecoding[k].DomainOut() > ndom)
+	  ndom = facedecoding[k].DomainOut();
+      }
 
+    return ndom;
+  }
 
 
-void Mesh :: SurfaceMeshOrientation ()
-{
-  int i, j;
-  int nse = GetNSE();
+
+  void Mesh :: SurfaceMeshOrientation ()
+  {
+    int i, j;
+    int nse = GetNSE();
   
-  BitArray used(nse);
-  used.Clear();
-  INDEX_2_HASHTABLE<int> edges(nse+1);
+    BitArray used(nse);
+    used.Clear();
+    INDEX_2_HASHTABLE<int> edges(nse+1);
 
-  bool haschanged = 0;
+    bool haschanged = 0;
 
 
-  const Element2d & tri = SurfaceElement(1);
-  for (j = 1; j <= 3; j++)
-    {
-      INDEX_2 i2(tri.PNumMod(j), tri.PNumMod(j+1));
-      edges.Set (i2, 1);
-    }
-  used.Set(1);
+    const Element2d & tri = SurfaceElement(1);
+    for (j = 1; j <= 3; j++)
+      {
+	INDEX_2 i2(tri.PNumMod(j), tri.PNumMod(j+1));
+	edges.Set (i2, 1);
+      }
+    used.Set(1);
 
-  bool unused;
-  do
-    {
-      bool changed;
-      do
-	{
-	  changed = 0;
-	  for (i = 1; i <= nse; i++)
-	    if (!used.Test(i))
-	      {
-		Element2d & el = surfelements.Elem(i);
-		int found = 0, foundrev = 0;
-		for (j = 1; j <= 3; j++)
-		  {
-		    INDEX_2 i2(el.PNumMod(j), el.PNumMod(j+1));
-		    if (edges.Used(i2))
-		      foundrev = 1;
-		    Swap (i2.I1(), i2.I2());
-		    if (edges.Used(i2))
-		      found = 1;
-		  }
+    bool unused;
+    do
+      {
+	bool changed;
+	do
+	  {
+	    changed = 0;
+	    for (i = 1; i <= nse; i++)
+	      if (!used.Test(i))
+		{
+		  Element2d & el = surfelements.Elem(i);
+		  int found = 0, foundrev = 0;
+		  for (j = 1; j <= 3; j++)
+		    {
+		      INDEX_2 i2(el.PNumMod(j), el.PNumMod(j+1));
+		      if (edges.Used(i2))
+			foundrev = 1;
+		      swap (i2.I1(), i2.I2());
+		      if (edges.Used(i2))
+			found = 1;
+		    }
 		
-		if (found || foundrev)
-		  {
-		    if (foundrev)
-		      Swap (el.PNum(2), el.PNum(3));
+		  if (found || foundrev)
+		    {
+		      if (foundrev)
+			swap (el.PNum(2), el.PNum(3));
 		    
-		    changed = 1;
-		    for (j = 1; j <= 3; j++)
-		      {
-			INDEX_2 i2(el.PNumMod(j), el.PNumMod(j+1));
-			edges.Set (i2, 1);
-		      }
-		    used.Set (i);
-		  }
-	      }
-	  if (changed)
-	    haschanged = 1;
-	}
-      while (changed);
+		      changed = 1;
+		      for (j = 1; j <= 3; j++)
+			{
+			  INDEX_2 i2(el.PNumMod(j), el.PNumMod(j+1));
+			  edges.Set (i2, 1);
+			}
+		      used.Set (i);
+		    }
+		}
+	    if (changed)
+	      haschanged = 1;
+	  }
+	while (changed);
       
 
-      unused = 0;
-      for (i = 1; i <= nse; i++)
-	if (!used.Test(i))
-	  {
-	    unused = 1;
-	    const Element2d & tri = SurfaceElement(i);
-	    for (j = 1; j <= 3; j++)
-	      {
-		INDEX_2 i2(tri.PNumMod(j), tri.PNumMod(j+1));
-		edges.Set (i2, 1);
-	      }
-	    used.Set(i);
-	    break;
-	  }
-    }
-  while (unused);
+	unused = 0;
+	for (i = 1; i <= nse; i++)
+	  if (!used.Test(i))
+	    {
+	      unused = 1;
+	      const Element2d & tri = SurfaceElement(i);
+	      for (j = 1; j <= 3; j++)
+		{
+		  INDEX_2 i2(tri.PNumMod(j), tri.PNumMod(j+1));
+		  edges.Set (i2, 1);
+		}
+	      used.Set(i);
+	      break;
+	    }
+      }
+    while (unused);
 
-  if (haschanged)
-    timestamp = NextTimeStamp();
-}
+    if (haschanged)
+      timestamp = NextTimeStamp();
+  }
   
 
-void Mesh :: Split2Tets()
-{
-  int oldne, oldnse;
-  int i, j, k, l;
+  void Mesh :: Split2Tets()
+  {
+    // int oldne, oldnse;
+    // int i, j, k, l;
   
-  oldne = GetNE(); 
-  for (i = 1; i <= oldne; i++)
-    {
-      Element el = VolumeElement(i);
-      if (el.GetType() == PRISM)
-	{
-	  //	  (*testout) << "split el: " << el << " to ";
+    bool has_prisms = 0;
 
-	  // prism, to 3 tets
-	  
-	  // make minimal node to node 1
-	  int minpi=0;
-	  PointIndex minpnum;
-	  minpnum = GetNP() + 1;
+    int oldne = GetNE(); 
+    for (int i = 1; i <= oldne; i++)
+      {
+	Element el = VolumeElement(i);
 
-	  for (j = 1; j <= 6; j++)
-	    {
-	      if (el.PNum(j) < minpnum)
-		{
-		  minpnum = el.PNum(j);
-		  minpi = j;
-		}
-	    }
+	if (el.GetType() == PRISM)
+	  {
+	    // prism, to 3 tets
 
-	  if (minpi >= 4)
-	    {
-	      for (j = 1; j <= 3; j++)
-		Swap (el.PNum(j), el.PNum(j+3));
-	      minpi -= 3;
-	    }
+	    // make minimal node to node 1
+	    int minpi=0;
+	    PointIndex minpnum;
+	    minpnum = GetNP() + 1;
 
-	  while (minpi > 1)
-	    {
-	      int hi = 0;
-	      for (j = 0; j <= 3; j+= 3)
-		{
-		  hi = el.PNum(1+j);
-		  el.PNum(1+j) = el.PNum(2+j);
-		  el.PNum(2+j) = el.PNum(3+j);
-		  el.PNum(3+j) = hi;
-		}
-	      minpi--;
-	    }
+	    for (int j = 1; j <= 6; j++)
+	      {
+		if (el.PNum(j) < minpnum)
+		  {
+		    minpnum = el.PNum(j);
+		    minpi = j;
+		  }
+	      }
+
+	    if (minpi >= 4)
+	      {
+		for (int j = 1; j <= 3; j++)
+		  swap (el.PNum(j), el.PNum(j+3));
+		minpi -= 3;
+	      }
 
-	  /*
-	    version 1: edge from pi2 to pi6,
-	    version 2: edge from pi3 to pi5,
-	   */
+	    while (minpi > 1)
+	      {
+		int hi = 0;
+		for (int j = 0; j <= 3; j+= 3)
+		  {
+		    hi = el.PNum(1+j);
+		    el.PNum(1+j) = el.PNum(2+j);
+		    el.PNum(2+j) = el.PNum(3+j);
+		    el.PNum(3+j) = hi;
+		  }
+		minpi--;
+	      }
 
-	  (*testout) << " rot el = " << el << " ";
+	    /*
+	      version 1: edge from pi2 to pi6,
+	      version 2: edge from pi3 to pi5,
+	    */
 
-	  static const int ntets[2][12] =
-	  { { 1, 4, 5, 6, 1, 2, 3, 6, 1, 2, 5, 6 },
-	    { 1, 4, 5, 6, 1, 2, 3, 5, 3, 1, 5, 6 } };
+	    static const int ntets[2][12] =
+	      { { 1, 4, 5, 6, 1, 2, 3, 6, 1, 2, 5, 6 },
+		{ 1, 4, 5, 6, 1, 2, 3, 5, 3, 1, 5, 6 } };
 
-	  const int * min2pi;
+	    const int * min2pi;
 
-	  if (min2 (el.PNum(2), el.PNum(6)) <
-	      min2 (el.PNum(3), el.PNum(5)))
-	    {
-	      min2pi = &ntets[0][0];
-	      (*testout) << "version 1 ";
-	    }
-	  else
-	    {
-	      min2pi = &ntets[1][0];
-	      (*testout) << "version 2 ";
-	    }
+	    if (min2 (el.PNum(2), el.PNum(6)) <
+		min2 (el.PNum(3), el.PNum(5)))
+	      {
+		min2pi = &ntets[0][0];
+		// (*testout) << "version 1 ";
+	      }
+	    else
+	      {
+		min2pi = &ntets[1][0];
+		// (*testout) << "version 2 ";
+	      }
 
 	  
-	  int firsttet = 1;
-	  for (j = 1; j <= 3; j++)
-	    {
-	      Element nel(4);
-	      for (k = 1; k <= 4; k++)
-		nel.PNum(k) = el.PNum(min2pi[4 * j + k - 5]);
-	      nel.SetIndex (el.GetIndex());
+	    int firsttet = 1;
+	    for (int j = 1; j <= 3; j++)
+	      {
+		Element nel(4);
+		for (int k = 1; k <= 4; k++)
+		  nel.PNum(k) = el.PNum(min2pi[4 * j + k - 5]);
+		nel.SetIndex (el.GetIndex());
+
+		int legal = 1;
+		for (int k = 1; k <= 3; k++)
+		  for (int l = k+1; l <= 4; l++)
+		    if (nel.PNum(k) == nel.PNum(l))
+		      legal = 0;
+
+		// (*testout) << nel << " ";
+		if (legal)
+		  {
+		    if (firsttet)
+		      {
+			VolumeElement(i) = nel;
+			firsttet = 0;
+		      }
+		    else
+		      {
+			AddVolumeElement(nel);
+		      }
+		  }
+	      }
+	    if (firsttet) (*testout) << "no legal";
+	    (*testout) << endl;
+	  }
+      
 
-	      int legal = 1;
-	      for (k = 1; k <= 3; k++)
-		for (l = k+1; l <= 4; l++)
-		  if (nel.PNum(k) == nel.PNum(l))
-		    legal = 0;
 
-	      (*testout) << nel << " ";
+	else if (el.GetType() == HEX)
+	  {
+	    // hex to A) 2 prisms or B) to 5 tets
 
-	      if (legal)
-		{
-		  if (firsttet)
-		    {
+	    // make minimal node to node 1
+	    int minpi=0;
+	    PointIndex minpnum;
+	    minpnum = GetNP() + 1;
+
+	    for (int j = 1; j <= 8; j++)
+	      {
+		if (el.PNum(j) < minpnum)
+		  {
+		    minpnum = el.PNum(j);
+		    minpi = j;
+		  }
+	      }
+
+	    if (minpi >= 5)
+	      {
+		for (int j = 1; j <= 4; j++)
+		  swap (el.PNum(j), el.PNum(j+4));
+		minpi -= 4;
+	      }
+
+	    while (minpi > 1)
+	      {
+		int hi = 0;
+		for (int j = 0; j <= 4; j+= 4)
+		  {
+		    hi = el.PNum(1+j);
+		    el.PNum(1+j) = el.PNum(2+j);
+		    el.PNum(2+j) = el.PNum(3+j);
+		    el.PNum(3+j) = el.PNum(4+j);
+		    el.PNum(4+j) = hi;
+		  }
+		minpi--;
+	      }
+
+
+
+	    static const int to_prisms[3][12] =
+	      { { 0, 1, 2, 4, 5, 6, 0, 2, 3, 4, 6, 7 },
+		{ 0, 1, 5, 3, 2, 6, 0, 5, 4, 3, 5, 7 },
+		{ 0, 7, 4, 1, 6, 5, 0, 3, 7, 1, 2, 6 },
+	      };
+
+	    const int * min2pi = 0;
+	    if (min2 (el[4], el[6]) < min2 (el[5], el[7]))
+	      min2pi = &to_prisms[0][0];
+	    else if (min2 (el[3], el[6]) < min2 (el[2], el[7]))
+	      min2pi = &to_prisms[1][0];
+	    else if (min2 (el[1], el[6]) < min2 (el[2], el[5]))
+	      min2pi = &to_prisms[2][0];
+
+	    if (min2pi)
+	      {
+		has_prisms = 1;
+		for (int j = 0; j < 2; j++)
+		  {
+		    Element nel(PRISM);
+		    for (int k = 0; k < 6; k++)
+		      nel[k] = el[min2pi[6*j + k]];
+		    nel.SetIndex (el.GetIndex());
+		  
+		    if (j == 0)
 		      VolumeElement(i) = nel;
-		      firsttet = 0;
-		    }
-		  else
-		    {
+		    else
 		      AddVolumeElement(nel);
-		    }
-		}
-	    }
-	  if (firsttet) (*testout) << "no legal";
-	  (*testout) << endl;
-	}
-      
+		  }
+	      }
+	    else
+	      {
+		// split to 5 tets
+	      
+		static const int to_tets[20] =
+		  {
+		    1, 2, 0, 5,
+		    3, 0, 2, 7,
+		    4, 5, 7, 0,
+		    6, 7, 5, 2,
+		    0, 2, 7, 5
+		  };
+
+		for (int j = 0; j < 5; j++)
+		  {
+		    Element nel(TET);
+		    for (int k = 0; k < 4; k++)
+		      nel[k] = el[to_tets[4*j + k]];
+		    nel.SetIndex (el.GetIndex());
+		  
+		    if (j == 0)
+		      VolumeElement(i) = nel;
+		    else
+		      AddVolumeElement(nel);
+		  }
+	      
+	      }
+	  }
       
-      if (el.GetType() == PYRAMID)
-	{
-	  // pyramid, to 2 tets
-	  
 
 
-	  static const int ntets[2][8] =
-	  { { 1, 2, 3, 5, 1, 3, 4, 5 },
-	    { 1, 2, 4, 5, 4, 2, 3, 5 }};
 
-	  const int * min2pi;
-
-	  if (min2 (el.PNum(1), el.PNum(3)) <
-	      min2 (el.PNum(2), el.PNum(4)))
-	    min2pi = &ntets[0][0];
-	  else
-	    min2pi = &ntets[1][0];
+      
+	else if (el.GetType() == PYRAMID)
+	  {
+	    // pyramid, to 2 tets
+	  
+	    static const int ntets[2][8] =
+	      { { 1, 2, 3, 5, 1, 3, 4, 5 },
+		{ 1, 2, 4, 5, 4, 2, 3, 5 }};
 
+	    const int * min2pi;
 
-	  int firsttet = 1;
-	  for (j = 1; j <= 2; j++)
-	    {
-	      Element nel(4);
-	      for (k = 1; k <= 4; k++)
-		nel.PNum(k) = el.PNum(min2pi[4 * j + k - 5]);
-	      nel.SetIndex (el.GetIndex());
+	    if (min2 (el.PNum(1), el.PNum(3)) <
+		min2 (el.PNum(2), el.PNum(4)))
+	      min2pi = &ntets[0][0];
+	    else
+	      min2pi = &ntets[1][0];
 
 
-	      int legal = 1;
-	      for (k = 1; k <= 3; k++)
-		for (l = k+1; l <= 4; l++)
-		  if (nel.PNum(k) == nel.PNum(l))
-		    legal = 0;
+	    int firsttet = 1;
+	    for (int j = 1; j <= 2; j++)
+	      {
+		Element nel(4);
+		for (int k = 1; k <= 4; k++)
+		  nel.PNum(k) = el.PNum(min2pi[4 * j + k - 5]);
+		nel.SetIndex (el.GetIndex());
 
-	      if (legal)
-		{
-		  (*testout) << nel << " ";
-		  if (firsttet)
-		    {
-		      VolumeElement(i) = nel;
-		      firsttet = 0;
-		    }
-		  else
-		    {
-		      AddVolumeElement(nel);
-		    }
-		}
-	    }
-	  (*testout) << endl;
 
+		int legal = 1;
+		for (int k = 1; k <= 3; k++)
+		  for (int l = k+1; l <= 4; l++)
+		    if (nel.PNum(k) == nel.PNum(l))
+		      legal = 0;
 
-	  
-	  
-	}
-    }
+		if (legal)
+		  {
+		    (*testout) << nel << " ";
+		    if (firsttet)
+		      {
+			VolumeElement(i) = nel;
+			firsttet = 0;
+		      }
+		    else
+		      {
+			AddVolumeElement(nel);
+		      }
+		  }
+	      }
+	    (*testout) << endl;
+	  }
+      }
 
   
-  oldnse = GetNSE(); 
-  for (i = 1; i <= oldnse; i++)
-    {
-      Element2d el = SurfaceElement(i);
-      if (el.GetNP() == 4)
-	{
-	  (*testout) << "split el: " << el << " to ";
+    int oldnse = GetNSE(); 
+    for (int i = 1; i <= oldnse; i++)
+      {
+	Element2d el = SurfaceElement(i);
+	if (el.GetNP() == 4)
+	  {
+	    (*testout) << "split el: " << el << " to ";
 	  
-	  static const int ntris[2][6] =
-	  { { 1, 2, 3, 1, 3, 4 },
-	    { 1, 2, 4, 4, 2, 3 }};
+	    static const int ntris[2][6] =
+	      { { 1, 2, 3, 1, 3, 4 },
+		{ 1, 2, 4, 4, 2, 3 }};
 
-	  const int * min2pi;
+	    const int * min2pi;
 
-	  if (min2 (el.PNum(1), el.PNum(3)) <
-	      min2 (el.PNum(2), el.PNum(4)))
-	    min2pi = &ntris[0][0];
-	  else
-	    min2pi = &ntris[1][0];
+	    if (min2 (el.PNum(1), el.PNum(3)) <
+		min2 (el.PNum(2), el.PNum(4)))
+	      min2pi = &ntris[0][0];
+	    else
+	      min2pi = &ntris[1][0];
 
-	  for (j = 0; j <6; j++)
-	    (*testout) << min2pi[j] << " ";
+	    for (int j = 0; j <6; j++)
+	      (*testout) << min2pi[j] << " ";
 
 
-	  int firsttri = 1;
-	  for (j = 1; j <= 2; j++)
-	    {
-	      Element2d nel(3);
-	      for (k = 1; k <= 3; k++)
-		nel.PNum(k) = el.PNum(min2pi[3 * j + k - 4]);
-	      nel.SetIndex (el.GetIndex());
-
-	      int legal = 1;
-	      for (k = 1; k <= 2; k++)
-		for (l = k+1; l <= 3; l++)
-		  if (nel.PNum(k) == nel.PNum(l))
-		    legal = 0;
-
-	      if (legal)
-		{
-		  (*testout) << nel << " ";
-		  if (firsttri)
-		    {
-		      SurfaceElement(i) = nel;
-		      firsttri = 0;
-		    }
-		  else
-		    {
-		      AddSurfaceElement(nel);
-		    }
-		}
-	    }
-	  (*testout) << endl;
+	    int firsttri = 1;
+	    for (int j = 1; j <= 2; j++)
+	      {
+		Element2d nel(3);
+		for (int k = 1; k <= 3; k++)
+		  nel.PNum(k) = el.PNum(min2pi[3 * j + k - 4]);
+		nel.SetIndex (el.GetIndex());
+
+		int legal = 1;
+		for (int k = 1; k <= 2; k++)
+		  for (int l = k+1; l <= 3; l++)
+		    if (nel.PNum(k) == nel.PNum(l))
+		      legal = 0;
+
+		if (legal)
+		  {
+		    (*testout) << nel << " ";
+		    if (firsttri)
+		      {
+			SurfaceElement(i) = nel;
+			firsttri = 0;
+		      }
+		    else
+		      {
+			AddSurfaceElement(nel);
+		      }
+		  }
+	      }
+	    (*testout) << endl;
 
-	}
-    }
+	  }
+      }
 
 
-  for (i = 1; i <= GetNE(); i++)
-    {
-      Element & el = VolumeElement(i);
-      const Point3d & p1 = Point (el.PNum(1));
-      const Point3d & p2 = Point (el.PNum(2));
-      const Point3d & p3 = Point (el.PNum(3));
-      const Point3d & p4 = Point (el.PNum(4));
-
-      double vol = (Vec3d (p1, p2) * 
-		    Cross (Vec3d (p1, p3), Vec3d(p1, p4)));
-      if (vol > 0)
-	Swap (el.PNum(3), el.PNum(4));
-    }
-  timestamp = NextTimeStamp();
-}
+    if (has_prisms)
 
-void Mesh :: BuildElementSearchTree ()
-{
-  if (elementsearchtreets == GetTimeStamp())
-    return;
+      Split2Tets();
+  
+    else
+      {
+	for (int i = 1; i <= GetNE(); i++)
+	  {
+	    Element & el = VolumeElement(i);
+	    const Point3d & p1 = Point (el.PNum(1));
+	    const Point3d & p2 = Point (el.PNum(2));
+	    const Point3d & p3 = Point (el.PNum(3));
+	    const Point3d & p4 = Point (el.PNum(4));
+	  
+	    double vol = (Vec3d (p1, p2) * 
+			  Cross (Vec3d (p1, p3), Vec3d(p1, p4)));
+	    if (vol > 0)
+	      swap (el.PNum(3), el.PNum(4));
+	  }
 
-  NgLock lock(mutex);
-  lock.Lock();
+	timestamp = NextTimeStamp();
+      }
+  }
 
-  PrintMessage (4, "Rebuild element searchtree");
+  void Mesh :: BuildElementSearchTree ()
+  {
+    if (elementsearchtreets == GetTimeStamp())
+      return;
 
-  if (elementsearchtree)
-    delete elementsearchtree;
-  elementsearchtree = NULL;
+    NgLock lock(mutex);
+    lock.Lock();
 
-  Box3d box;
-  int i, j;
-  int ne = GetNE();
-  if (!ne) 
-    {
-      lock.UnLock();
-      return;
-    }
+    PrintMessage (4, "Rebuild element searchtree");
 
-  box.SetPoint (Point (VolumeElement(1).PNum(1)));
-  for (i = 1; i <= ne; i++)
-    {
-      const Element & el = VolumeElement(i);
-      for (j = 1; j <= el.GetNP(); j++)
-	box.AddPoint (Point (el.PNum(j)));
-    }
+    if (elementsearchtree)
+      delete elementsearchtree;
+    elementsearchtree = NULL;
+
+    Box3d box;
+    int i, j;
+    int ne = GetNE();
+    if (!ne) 
+      {
+	lock.UnLock();
+	return;
+      }
+
+    box.SetPoint (Point (VolumeElement(1).PNum(1)));
+    for (i = 1; i <= ne; i++)
+      {
+	const Element & el = VolumeElement(i);
+	for (j = 1; j <= el.GetNP(); j++)
+	  box.AddPoint (Point (el.PNum(j)));
+      }
   
-  box.Increase (1.01 * box.CalcDiam());
-  elementsearchtree = new Box3dTree (box.PMin(), box.PMax());
+    box.Increase (1.01 * box.CalcDiam());
+    elementsearchtree = new Box3dTree (box.PMin(), box.PMax());
   
 
 
-  for (i = 1; i <= ne; i++)
-    {
-      const Element & el = VolumeElement(i);
-      box.SetPoint (Point (el.PNum(1)));
-      for (j = 1; j <= el.GetNP(); j++)
-	box.AddPoint (Point (el.PNum(j)));
+    for (i = 1; i <= ne; i++)
+      {
+	const Element & el = VolumeElement(i);
+	box.SetPoint (Point (el.PNum(1)));
+	for (j = 1; j <= el.GetNP(); j++)
+	  box.AddPoint (Point (el.PNum(j)));
 
-      elementsearchtree -> Insert (box.PMin(), box.PMax(), i);
-    }
+	elementsearchtree -> Insert (box.PMin(), box.PMax(), i);
+      }
 
-  elementsearchtreets = GetTimeStamp();
+    elementsearchtreets = GetTimeStamp();
 
-  lock.UnLock();
-}
+    lock.UnLock();
+  }
 
 
-int Mesh :: GetElementOfPoint (const Point3d & p,
-			       double lami[3]) const
-{
-  if (dimension == 2)
-    {
-      int i, j;
-      Vec3d col1, col2, col3;
-      Vec3d rhs, sol;
-      double eps = 1e-6;
-      int ne;
-      
-      ARRAY<int> locels;
-      if (0)
-	{
-	  elementsearchtree->GetIntersecting (p, p, locels);
-	  ne = locels.Size();
-	}
-      else
-	ne = GetNSE();
+  int Mesh :: GetElementOfPoint (const Point3d & p,
+				 double lami[3],
+				 bool build_searchtree,
+				 const int index) const
+  {
+    if (dimension == 2)
+      {
+	int i, j;
+	Vec3d col1, col2, col3;
+	Vec3d rhs, sol;
+	double eps = 1e-6;
+	int ne;
       
-      ARRAY<Element2d> loctrigs;
-      Vec3d nv(0, 0, 1);
+	ARRAY<int> locels;
+	if (0)
+	  {
+	    elementsearchtree->GetIntersecting (p, p, locels);
+	    ne = locels.Size();
+	  }
+	else
+	  ne = GetNSE();
+	ARRAY<Element2d> loctrigs;
+	Vec3d nv(0, 0, 1);
+	for (i = 1; i <= ne; i++)
+	  {
+	    int ii;
+	    if (0)
+	      ii = locels.Get(i);
+	    else
+	      ii = i;
 
-      for (i = 1; i <= ne; i++)
-	{
-	  int ii;
-	  if (0)
-	    ii = locels.Get(i);
-	  else
-	    ii = i;
-	  
-	  //	  SurfaceElement(ii).GetTets (loctets);
-	  loctrigs.SetSize(1);
-	  loctrigs.Elem(1) = SurfaceElement(ii);
+	    if((index >= 0) && (index != SurfaceElement(ii).GetIndex())) continue;
 
-	  for (j = 1; j <= loctrigs.Size(); j++)
-	    {
-	      const Element2d & el = loctrigs.Get(j);
+	    //SZ 
+	    if(SurfaceElement(ii).GetType()==QUAD)
+	      {
+
+		const Element2d & el = SurfaceElement(ii); 
+	      
+		const Point3d & p1 = Point(el.PNum(1)); 
+		const Point3d & p2 = Point(el.PNum(2));
+		const Point3d & p3 = Point(el.PNum(3));
+		const Point3d & p4 = Point(el.PNum(4)); 
 	      
-	      const Point3d & p1 = Point(el.PNum(1));
-	      const Point3d & p2 = Point(el.PNum(2));
-	      const Point3d & p3 = Point(el.PNum(3));
+		// Coefficients of Bilinear Mapping from Ref-Elem to global Elem
+		// X = a + b x + c y + d x y 
+		Vec3d a = p1; 
+		Vec3d b = p2 - a; 
+		Vec3d c = p4 - a; 
+		Vec3d d = p3 - a - b - c; 
 	      
-	      /*
-	      Box3d box;
-	      box.SetPoint (p1);
-	      box.AddPoint (p2);
-	      box.AddPoint (p3);
-	      box.AddPoint (p4);
-	      if (!box.IsIn (p))
-		continue;
-	      */
-	      col1 = p2-p1;
-	      col2 = p3-p1;
-	      col3 = nv;
-	      rhs = p - p1;
+		double dxb = d.X()*b.Y()-d.Y()*b.X();
+		double dxc = d.X()*c.Y()-d.Y()*c.X(); 
+		double dxa = d.X()*a.Y()-d.Y()*a.X(); 
+		double dxp = d.X()*p.Y()-d.Y()*p.X(); 
+	      	      
+		double c0,c1,c2,rt; 
+		lami[2]=0.; 
+		double eps = 1.E-12; 
+
+		if(fabs(d.X()) <= eps && fabs(d.Y())<= eps)
+		  {
+		    //Solve Linear System
+		    lami[0]=(c.Y()*(p.X()-a.X())-b.Y()*(p.Y()-a.Y()))/
+		      (b.X()*c.Y() -b.Y()*c.X()); 
+		    lami[1]=(-c.X()*(p.X()-a.X())+b.X()*(p.Y()-a.Y()))/
+		      (b.X()*c.Y() -b.Y()*c.X()); 
+		  } 
+		else
+		  if(fabs(dxb) <= eps) 
+		    {
+		      lami[1] = (dxp-dxa)/dxc;
+		      if(fabs(b.X()-d.X()*lami[1])>=eps)
+			lami[0] = (p.X()-a.X() - c.X()*lami[1])/(b.X()+d.X()*lami[1]); 
+		      else
+			lami[0] = (p.Y()-a.Y() - c.Y()*lami[1])/(b.Y()+d.Y()*lami[1]); 
+		    }
+		  else
+		    if(fabs(dxc) <= eps)
+		      {
+			lami[0] = (dxp-dxa)/dxb;
+			if(fabs(c.X()-d.X()*lami[0])>=eps)
+			  lami[1] = (p.X()-a.X() - b.X()*lami[0])/(c.X()+d.X()*lami[0]); 
+			else
+			  lami[1] = (p.Y()-a.Y() - b.Y()*lami[0])/(c.Y()+d.Y()*lami[0]); 
+		      }
+		    else //Solve quadratic equation
+		      {
+			if(fabs(d.X()) >= eps)
+			  {
+			    c2 = d.X()*dxc;
+			    c1 = d.X()*dxc - c.X()*dxb - d.X()*(dxp-dxa);
+			    c0 = -b.X()*(dxp -dxa) - (a.X()-p.X())*dxb;
+			  }
+			else 
+			  {
+			    c2 = d.Y()*dxc;
+			    c1 = d.Y()*dxc - c.Y()*dxb - d.Y()*(dxp-dxa);
+			    c0 = -b.Y()*(dxp -dxa) - (a.Y()-p.Y())*dxb;
+			  }
+
+			double rt =  c1*c1 - 4*c2*c0;
+			if (rt < 0.) continue; 
+			lami[1] = (-c1 + sqrt(rt))/2/c2;
+			if(lami[1]<=1. && lami[1]>=0.)
+			  {
+			    lami[0] = (dxp - dxa -dxc*lami[1])/dxb;
+			    if(lami[0]<=1. && lami[0]>=0.)
+			      return(ii);
+			  }
+		      
+			lami[1] = (-c1 - sqrt(rt))/2/c2;
+			lami[0] = (dxp - dxa -dxc*lami[1])/dxb;
+		      }
 	      
-	      SolveLinearSystem (col1, col2, col3, rhs, sol);
+		if( lami[0] <= 1.+eps  && lami[0] >= -eps && lami[1]<=1.+eps && lami[1]>=-eps)
+		  return(ii);
+		      
+		continue; 
 	      
-	      if (sol.X() >= -eps && sol.Y() >= -eps && 
-		  sol.X() + sol.Y() <= 1+eps)
-		{
-		  lami[0] = sol.X();
-		  lami[1] = sol.Y();
-		  lami[2] = sol.Z();
-		  return ii;
-		}
-	    }
-	}
-      
-      return 0;
-    }
-
-  else
+	      }
+	    else
+	      {
+		//	  SurfaceElement(ii).GetTets (loctets);
+		loctrigs.SetSize(1);
+		loctrigs.Elem(1) = SurfaceElement(ii);
+	      
+	      
+	      
+		for (j = 1; j <= loctrigs.Size(); j++)
+		  {
+		    const Element2d & el = loctrigs.Get(j);
+		  
+		  
+		    const Point3d & p1 = Point(el.PNum(1));
+		    const Point3d & p2 = Point(el.PNum(2));
+		    const Point3d & p3 = Point(el.PNum(3));
+		    /*
+		      Box3d box;
+		      box.SetPoint (p1);
+		      box.AddPoint (p2);
+		      box.AddPoint (p3);
+		      box.AddPoint (p4);
+		      if (!box.IsIn (p))
+		      continue;
+		    */
+		    col1 = p2-p1;
+		    col2 = p3-p1;
+		    col3 = nv;
+		    rhs = p - p1;
+		  
+		    SolveLinearSystem (col1, col2, col3, rhs, sol);
+		  
+		    if (sol.X() >= -eps && sol.Y() >= -eps && 
+			sol.X() + sol.Y() <= 1+eps)
+		      {
+			lami[0] = sol.X();
+			lami[1] = sol.Y();
+			lami[2] = sol.Z();
+		      
+			return ii;
+		      }
+		  }
+	      }
+	  }
+	 
+	return 0;
+      }  
+    else
       
-    {
-      int i, j;
-      Vec3d col1, col2, col3;
-      Vec3d rhs, sol;
-      double eps = 1e-4;
-      int ne;
+      {
+	int i, j;
+	Vec3d col1, col2, col3;
+	Vec3d rhs, sol;
+	double eps = 1e-4;
+	int ne;
       
-      ARRAY<int> locels;
-      if (elementsearchtree)
-	{
-	  // update if necessary:
-	  const_cast<Mesh&>(*this).BuildElementSearchTree (); 
-	  elementsearchtree->GetIntersecting (p, p, locels);
-	  ne = locels.Size();
-	}
-      else
-	ne = GetNE();
+	ARRAY<int> locels;
+	if (elementsearchtree || build_searchtree)
+	  {
+	    // update if necessary:
+	    const_cast<Mesh&>(*this).BuildElementSearchTree (); 
+	    elementsearchtree->GetIntersecting (p, p, locels);
+	    ne = locels.Size();
+	  }
+	else
+	  ne = GetNE();
       
-      ARRAY<Element> loctets;
-      for (i = 1; i <= ne; i++)
-	{
-	  int ii;
-	  if (elementsearchtree)
-	    ii = locels.Get(i);
-	  else
-	    ii = i;
+	ARRAY<Element> loctets;
+	for (i = 1; i <= ne; i++)
+	  {
+	    int ii;
+	    if (elementsearchtree)
+	      ii = locels.Get(i);
+	    else
+	      ii = i;
 	  
-	  VolumeElement(ii).GetTets (loctets);
+	    VolumeElement(ii).GetTets (loctets);
 	  
-	  for (j = 1; j <= loctets.Size(); j++)
-	    {
-	      const Element & el = loctets.Get(j);
+	    for (j = 1; j <= loctets.Size(); j++)
+	      {
+		const Element & el = loctets.Get(j);
 	      
-	      const Point3d & p1 = Point(el.PNum(1));
-	      const Point3d & p2 = Point(el.PNum(2));
-	      const Point3d & p3 = Point(el.PNum(3));
-	      const Point3d & p4 = Point(el.PNum(4));
+		const Point3d & p1 = Point(el.PNum(1));
+		const Point3d & p2 = Point(el.PNum(2));
+		const Point3d & p3 = Point(el.PNum(3));
+		const Point3d & p4 = Point(el.PNum(4));
 	      
-	      Box3d box;
-	      box.SetPoint (p1);
-	      box.AddPoint (p2);
-	      box.AddPoint (p3);
-	      box.AddPoint (p4);
-	      if (!box.IsIn (p))
-		continue;
+		Box3d box;
+		box.SetPoint (p1);
+		box.AddPoint (p2);
+		box.AddPoint (p3);
+		box.AddPoint (p4);
+		if (!box.IsIn (p))
+		  continue;
 	      
-	      col1 = p2-p1;
-	      col2 = p3-p1;
-	      col3 = p4-p1;
-	      rhs = p - p1;
+		col1 = p2-p1;
+		col2 = p3-p1;
+		col3 = p4-p1;
+		rhs = p - p1;
 	      
-	      SolveLinearSystem (col1, col2, col3, rhs, sol);
+		SolveLinearSystem (col1, col2, col3, rhs, sol);
 	      
-	      if (sol.X() >= -eps && sol.Y() >= -eps && sol.Z() >= -eps &&
-		  sol.X() + sol.Y() + sol.Z() <= 1+eps)
-		{
-		  ARRAY<Element> loctetsloc;
-		  ARRAY<Point3d> pointsloc;
+		if (sol.X() >= -eps && sol.Y() >= -eps && sol.Z() >= -eps &&
+		    sol.X() + sol.Y() + sol.Z() <= 1+eps)
+		  {
+		    ARRAY<Element> loctetsloc;
+		    ARRAY<Point3d> pointsloc;
 
-		  VolumeElement(ii).GetTetsLocal (loctetsloc);
-		  VolumeElement(ii).GetNodesLocalNew (pointsloc);
+		    VolumeElement(ii).GetTetsLocal (loctetsloc);
+		    VolumeElement(ii).GetNodesLocalNew (pointsloc);
 
-		  const Element & le = loctetsloc.Get(j);
+		    const Element & le = loctetsloc.Get(j);
 		  
-		  Point3d p = 
-		    pointsloc.Get(le.PNum(1)) 
-		    + sol.X() * Vec3d (pointsloc.Get(le.PNum(1)), pointsloc.Get(le.PNum(2))) 
-		    + sol.Y() * Vec3d (pointsloc.Get(le.PNum(1)), pointsloc.Get(le.PNum(3))) 
-		    + sol.Z() * Vec3d (pointsloc.Get(le.PNum(1)), pointsloc.Get(le.PNum(4))) ;
+		    Point3d p = 
+		      pointsloc.Get(le.PNum(1)) 
+		      + sol.X() * Vec3d (pointsloc.Get(le.PNum(1)), pointsloc.Get(le.PNum(2))) 
+		      + sol.Y() * Vec3d (pointsloc.Get(le.PNum(1)), pointsloc.Get(le.PNum(3))) 
+		      + sol.Z() * Vec3d (pointsloc.Get(le.PNum(1)), pointsloc.Get(le.PNum(4))) ;
 
 
-		  lami[0] = p.X();
-		  lami[1] = p.Y();
-		  lami[2] = p.Z();
-		  return ii;
-		}
-	    }
-	}
+		    lami[0] = p.X();
+		    lami[1] = p.Y();
+		    lami[2] = p.Z();
+		    return ii;
+		  }
+	      }
+	  }
       
-      return 0;
-    }
-}
+	return 0;
+      }
+  }
 
 
-void Mesh::GetIntersectingVolEls(const Point3d& p1, const Point3d& p2, 
-				 ARRAY<int> & locels) const
-{
-  elementsearchtree->GetIntersecting (p1, p2, locels);
-}
+  void Mesh::GetIntersectingVolEls(const Point3d& p1, const Point3d& p2, 
+				   ARRAY<int> & locels) const
+  {
+    elementsearchtree->GetIntersecting (p1, p2, locels);
+  }
 
-void Mesh :: SplitIntoParts()
-{
-  int i, j, dom;
-  int ne = GetNE();
-  int np = GetNP();
-  int nse = GetNSE();
+  void Mesh :: SplitIntoParts()
+  {
+    int i, j, dom;
+    int ne = GetNE();
+    int np = GetNP();
+    int nse = GetNSE();
 
-  BitArray surfused(nse);
-  BitArray pused (np);
+    BitArray surfused(nse);
+    BitArray pused (np);
 
-  surfused.Clear();
+    surfused.Clear();
 
-  dom = 0;
+    dom = 0;
   
-  while (1)
-    {
-      int cntd = 1;
+    while (1)
+      {
+	int cntd = 1;
 
-      dom++;
+	dom++;
       
-      pused.Clear();
-
-      int found = 0;
-      for (i = 1; i <= nse; i++)
-	if (!surfused.Test(i))
-	  {
-	    SurfaceElement(i).SetIndex (dom);
-	    for (j = 1; j <= 3; j++)
-	      pused.Set (SurfaceElement(i).PNum(j));
-	    found = 1;
-	    cntd = 1;
-	    surfused.Set(i);
-	    break;
-	  }
-
-      if (!found)
-	break;
+	pused.Clear();
 
-      int change;
-      do
-	{
-	  change = 0;
-	  for (i = 1; i <= nse; i++)
+	int found = 0;
+	for (i = 1; i <= nse; i++)
+	  if (!surfused.Test(i))
 	    {
-	      int is = 0, isnot = 0;
+	      SurfaceElement(i).SetIndex (dom);
 	      for (j = 1; j <= 3; j++)
-		if (pused.Test(SurfaceElement(i).PNum(j)))
-		  is = 1;
-		else
-		  isnot = 1;
+		pused.Set (SurfaceElement(i).PNum(j));
+	      found = 1;
+	      cntd = 1;
+	      surfused.Set(i);
+	      break;
+	    }
+
+	if (!found)
+	  break;
+
+	int change;
+	do
+	  {
+	    change = 0;
+	    for (i = 1; i <= nse; i++)
+	      {
+		int is = 0, isnot = 0;
+		for (j = 1; j <= 3; j++)
+		  if (pused.Test(SurfaceElement(i).PNum(j)))
+		    is = 1;
+		  else
+		    isnot = 1;
 	      
-	      if (is && isnot)
-		{
-		  change = 1;
-		  for (j = 1; j <= 3; j++)
-		    pused.Set (SurfaceElement(i).PNum(j));
-		}
+		if (is && isnot)
+		  {
+		    change = 1;
+		    for (j = 1; j <= 3; j++)
+		      pused.Set (SurfaceElement(i).PNum(j));
+		  }
 
-	      if (is) 
-		{
-		  if (!surfused.Test(i))
-		    {
-		      surfused.Set(i);
-		      SurfaceElement(i).SetIndex (dom);
-		      cntd++;
-		    }
-		}
-	    }
+		if (is) 
+		  {
+		    if (!surfused.Test(i))
+		      {
+			surfused.Set(i);
+			SurfaceElement(i).SetIndex (dom);
+			cntd++;
+		      }
+		  }
+	      }
 
 
-	  for (i = 1; i <= ne; i++)
-	    {
-	      int is = 0, isnot = 0;
-	      for (j = 1; j <= 4; j++)
-		if (pused.Test(VolumeElement(i).PNum(j)))
-		  is = 1;
-		else
-		  isnot = 1;
+	    for (i = 1; i <= ne; i++)
+	      {
+		int is = 0, isnot = 0;
+		for (j = 1; j <= 4; j++)
+		  if (pused.Test(VolumeElement(i).PNum(j)))
+		    is = 1;
+		  else
+		    isnot = 1;
 	      
-	      if (is && isnot)
-		{
-		  change = 1;
-		  for (j = 1; j <= 4; j++)
-		    pused.Set (VolumeElement(i).PNum(j));
-		}
+		if (is && isnot)
+		  {
+		    change = 1;
+		    for (j = 1; j <= 4; j++)
+		      pused.Set (VolumeElement(i).PNum(j));
+		  }
 
-	      if (is)
-		{
-		  VolumeElement(i).SetIndex (dom);
-		}
-	    }
-	}
-      while (change);
+		if (is)
+		  {
+		    VolumeElement(i).SetIndex (dom);
+		  }
+	      }
+	  }
+	while (change);
 
-      PrintMessage (3, "domain ", dom, " has ", cntd, " surfaceelements");
-    }
+	PrintMessage (3, "domain ", dom, " has ", cntd, " surfaceelements");
+      }
 
-  /*
-  facedecoding.SetSize (dom);
-  for (i = 1; i <= dom; i++)
-    {
+    /*
+      facedecoding.SetSize (dom);
+      for (i = 1; i <= dom; i++)
+      {
       facedecoding.Elem(i).surfnr = 0;
       facedecoding.Elem(i).domin = i;
       facedecoding.Elem(i).domout = 0;
-    }
+      }
     */
-  ClearFaceDescriptors();
-  for (i = 1; i <= dom; i++)
-    AddFaceDescriptor (FaceDescriptor (0, i, 0, 0));
-  CalcSurfacesOfNode();
-  timestamp = NextTimeStamp();
-}
+    ClearFaceDescriptors();
+    for (i = 1; i <= dom; i++)
+      AddFaceDescriptor (FaceDescriptor (0, i, 0, 0));
+    CalcSurfacesOfNode();
+    timestamp = NextTimeStamp();
+  }
 
-void Mesh :: SplitSeparatedFaces ()
-{
-  int fdi;
-  int i, j;
-  int np = GetNP();
+  void Mesh :: SplitSeparatedFaces ()
+  {
+    int fdi;
+    int i, j;
+    int np = GetNP();
 
-  BitArray usedp(np);
+    BitArray usedp(np);
 
-  fdi = 1;
-  while (fdi <= GetNFD())
-    {
-      int firstel = 0;
-      for (i = 1; i <= GetNSE(); i++)
-	if (SurfaceElement(i).GetIndex() == fdi)
-	  {
-	    firstel = i;
-	    break;
-	  }
-      if (!firstel) continue;
+    fdi = 1;
+    while (fdi <= GetNFD())
+      {
+	int firstel = 0;
+	for (i = 1; i <= GetNSE(); i++)
+	  if (SurfaceElement(i).GetIndex() == fdi)
+	    {
+	      firstel = i;
+	      break;
+	    }
+	if (!firstel) continue;
 
-      usedp.Clear();
-      for (j = 1; j <= SurfaceElement(firstel).GetNP(); j++)
-	usedp.Set (SurfaceElement(firstel).PNum(j));
+	usedp.Clear();
+	for (j = 1; j <= SurfaceElement(firstel).GetNP(); j++)
+	  usedp.Set (SurfaceElement(firstel).PNum(j));
 
-      int changed;
-      do
-	{
-	  changed = 0;
-	  for (i = 1; i <= GetNSE(); i++)
-	    {
-	      const Element2d & el = SurfaceElement(i);
-	      if (el.GetIndex() != fdi)
-		continue;
+	int changed;
+	do
+	  {
+	    changed = 0;
+	    for (i = 1; i <= GetNSE(); i++)
+	      {
+		const Element2d & el = SurfaceElement(i);
+		if (el.GetIndex() != fdi)
+		  continue;
 
-	      int has = 0;
-	      int hasno = 0;
-	      for (j = 1; j <= el.GetNP(); j++)
-		{
-		  if (usedp.Test(el.PNum(j)))
-		    has = 1;
-		  else
-		    hasno = 1;
-		}
-	      if (has && hasno)
-		changed = 1;
+		int has = 0;
+		int hasno = 0;
+		for (j = 1; j <= el.GetNP(); j++)
+		  {
+		    if (usedp.Test(el.PNum(j)))
+		      has = 1;
+		    else
+		      hasno = 1;
+		  }
+		if (has && hasno)
+		  changed = 1;
 
-	      if (has)
-		for (j = 1; j <= el.GetNP(); j++)
-		  usedp.Set (el.PNum(j));
-	    }
-	}
-      while (changed);
+		if (has)
+		  for (j = 1; j <= el.GetNP(); j++)
+		    usedp.Set (el.PNum(j));
+	      }
+	  }
+	while (changed);
 
-      int nface = 0;
-      for (i = 1; i <= GetNSE(); i++)
-	{
-	  Element2d & el = SurfaceElement(i);
-	  if (el.GetIndex() != fdi)
-	    continue;	  
+	int nface = 0;
+	for (i = 1; i <= GetNSE(); i++)
+	  {
+	    Element2d & el = SurfaceElement(i);
+	    if (el.GetIndex() != fdi)
+	      continue;	  
 
-	  int hasno = 0;
-	  for (j = 1; j <= el.GetNP(); j++)
-	    {
-	      if (!usedp.Test(el.PNum(j)))
-		hasno = 1;
-	    }
+	    int hasno = 0;
+	    for (j = 1; j <= el.GetNP(); j++)
+	      {
+		if (!usedp.Test(el.PNum(j)))
+		  hasno = 1;
+	      }
 	  
-	  if (hasno)
-	    {
-	      if (!nface)
-		{
-		  FaceDescriptor nfd = GetFaceDescriptor(fdi);
-		  nface = AddFaceDescriptor (nfd);
-		}
+	    if (hasno)
+	      {
+		if (!nface)
+		  {
+		    FaceDescriptor nfd = GetFaceDescriptor(fdi);
+		    nface = AddFaceDescriptor (nfd);
+		  }
 
-	      el.SetIndex (nface);
-	    }
-	}
-      fdi++;
-    }
-}
+		el.SetIndex (nface);
+	      }
+	  }
+	fdi++;
+      }
+  }
 
 
-void Mesh :: GetSurfaceElementsOfFace (int facenr, ARRAY<SurfaceElementIndex> & sei) const
-{
-  sei.SetSize (0);
-  for (SurfaceElementIndex i = 0; i < GetNSE(); i++)
-    if ( (*this)[i].GetIndex () == facenr && (*this)[i][0] >= PointIndex::BASE )
-      sei.Append (i);
-}
+  void Mesh :: GetSurfaceElementsOfFace (int facenr, ARRAY<SurfaceElementIndex> & sei) const
+  {
+    sei.SetSize (0);
+    for (SurfaceElementIndex i = 0; i < GetNSE(); i++)
+      if ( (*this)[i].GetIndex () == facenr && (*this)[i][0] >= PointIndex::BASE &&
+	   !(*this)[i].IsDeleted() )
+	sei.Append (i);
+  }
 
 
 
 
-void Mesh :: CalcMinMaxAngle (double badellimit, double * retvalues) 
-{
-  int i, j;
-  int lpi1, lpi2, lpi3, lpi4;
-  double phimax = 0, phimin = 10;
-  double facephimax = 0, facephimin = 10;
-  int illegaltets = 0, negativetets = 0, badtets = 0;
+  void Mesh :: CalcMinMaxAngle (double badellimit, double * retvalues) 
+  {
+    int i, j;
+    int lpi1, lpi2, lpi3, lpi4;
+    double phimax = 0, phimin = 10;
+    double facephimax = 0, facephimin = 10;
+    int illegaltets = 0, negativetets = 0, badtets = 0;
 
-  for (i = 1; i <= GetNE(); i++)
-    {
-      int badel = 0;
+    for (i = 1; i <= GetNE(); i++)
+      {
+	int badel = 0;
 
-      Element & el = VolumeElement(i);
+	Element & el = VolumeElement(i);
 
-      if (el.GetType() != TET)
-	{
-	  VolumeElement(i).flags.badel = 0;
-	  continue;
-	}
+	if (el.GetType() != TET)
+	  {
+	    VolumeElement(i).flags.badel = 0;
+	    continue;
+	  }
 
-      if (el.Volume(Points()) < 0)
-	{
-	  badel = 1;
-	  negativetets++;
-	}
+	if (el.Volume(Points()) < 0)
+	  {
+	    badel = 1;
+	    negativetets++;
+	  }
       
 
-      if (!LegalTet (el)) 
-	{
-	  badel = 1;
-	  illegaltets++;
-	  (*testout) << "illegal tet: " << i << " ";
-	  for (j = 1; j <= el.GetNP(); j++)
-	    (*testout) << el.PNum(j) << " ";
-	  (*testout) << endl;
-	}
-      
-	
-      // angles between faces
-      for (lpi1 = 1; lpi1 <= 3; lpi1++)
-	for (lpi2 = lpi1+1; lpi2 <= 4; lpi2++)
+	if (!LegalTet (el)) 
 	  {
-	    lpi3 = 1;
-	    while (lpi3 == lpi1 || lpi3 == lpi2)
-	      lpi3++;
-	    lpi4 = 10 - lpi1 - lpi2 - lpi3;
-
-	    const Point3d & p1 = Point (el.PNum(lpi1));
-	    const Point3d & p2 = Point (el.PNum(lpi2));
-	    const Point3d & p3 = Point (el.PNum(lpi3));
-	    const Point3d & p4 = Point (el.PNum(lpi4));
-
-	    Vec3d n(p1, p2);
-	    n /= n.Length();
-	    Vec3d v1(p1, p3);
-	    Vec3d v2(p1, p4);
-
-	    v1 -= (n * v1) * n;
-	    v2 -= (n * v2) * n;
-
-	    double cosphi = (v1 * v2) / (v1.Length() * v2.Length());
-	    double phi = acos (cosphi);
-	    if (phi > phimax) phimax = phi;
-	    if (phi < phimin) phimin = phi;
-
-	    if ((180/M_PI) * phi > badellimit)
-	      badel = 1;
+	    badel = 1;
+	    illegaltets++;
+	    (*testout) << "illegal tet: " << i << " ";
+	    for (j = 1; j <= el.GetNP(); j++)
+	      (*testout) << el.PNum(j) << " ";
+	    (*testout) << endl;
 	  }
-
-
-      // angles in faces
-      for (j = 1; j <= 4; j++)
-	{
-	  Element2d face;
-	  el.GetFace (j, face);
-	  for (lpi1 = 1; lpi1 <= 3; lpi1++)
+      
+	
+	// angles between faces
+	for (lpi1 = 1; lpi1 <= 3; lpi1++)
+	  for (lpi2 = lpi1+1; lpi2 <= 4; lpi2++)
 	    {
-	      lpi2 = lpi1 % 3 + 1;
-	      lpi3 = lpi2 % 3 + 1;
+	      lpi3 = 1;
+	      while (lpi3 == lpi1 || lpi3 == lpi2)
+		lpi3++;
+	      lpi4 = 10 - lpi1 - lpi2 - lpi3;
 
 	      const Point3d & p1 = Point (el.PNum(lpi1));
 	      const Point3d & p2 = Point (el.PNum(lpi2));
 	      const Point3d & p3 = Point (el.PNum(lpi3));
+	      const Point3d & p4 = Point (el.PNum(lpi4));
+
+	      Vec3d n(p1, p2);
+	      n /= n.Length();
+	      Vec3d v1(p1, p3);
+	      Vec3d v2(p1, p4);
+
+	      v1 -= (n * v1) * n;
+	      v2 -= (n * v2) * n;
 
-	      Vec3d v1(p1, p2);
-	      Vec3d v2(p1, p3);
 	      double cosphi = (v1 * v2) / (v1.Length() * v2.Length());
 	      double phi = acos (cosphi);
-	      if (phi > facephimax) facephimax = phi;
-	      if (phi < facephimin) facephimin = phi;
+	      if (phi > phimax) phimax = phi;
+	      if (phi < phimin) phimin = phi;
 
 	      if ((180/M_PI) * phi > badellimit)
 		badel = 1;
-
 	    }
-	}
+
+
+	// angles in faces
+	for (j = 1; j <= 4; j++)
+	  {
+	    Element2d face;
+	    el.GetFace (j, face);
+	    for (lpi1 = 1; lpi1 <= 3; lpi1++)
+	      {
+		lpi2 = lpi1 % 3 + 1;
+		lpi3 = lpi2 % 3 + 1;
+
+		const Point3d & p1 = Point (el.PNum(lpi1));
+		const Point3d & p2 = Point (el.PNum(lpi2));
+		const Point3d & p3 = Point (el.PNum(lpi3));
+
+		Vec3d v1(p1, p2);
+		Vec3d v2(p1, p3);
+		double cosphi = (v1 * v2) / (v1.Length() * v2.Length());
+		double phi = acos (cosphi);
+		if (phi > facephimax) facephimax = phi;
+		if (phi < facephimin) facephimin = phi;
+
+		if ((180/M_PI) * phi > badellimit)
+		  badel = 1;
+
+	      }
+	  }
 
        
-      VolumeElement(i).flags.badel = badel;
-      if (badel) badtets++;
-    }
+	VolumeElement(i).flags.badel = badel;
+	if (badel) badtets++;
+      }
 
-  if (!GetNE())
-    {
-      phimin = phimax = facephimin = facephimax = 0;
-    }
+    if (!GetNE())
+      {
+	phimin = phimax = facephimin = facephimax = 0;
+      }
 
-  if (!retvalues)
-    {
-      PrintMessage (1, "");
-      PrintMessage (1, "between planes:  phimin = ", (180/M_PI) * phimin,
-		    " phimax = ", (180/M_PI) *phimax);
-      PrintMessage (1, "inside planes:   phimin = ", (180/M_PI) * facephimin,
-		    " phimax = ", (180/M_PI) * facephimax);
-      PrintMessage (1, "");      
-    }
-  else
-    {
-      retvalues[0] = (180/M_PI) * facephimin;
-      retvalues[1] = (180/M_PI) * facephimax;
-      retvalues[2] = (180/M_PI) * phimin;
-      retvalues[3] = (180/M_PI) * phimax;
-    }
-  PrintMessage (3, "negative tets: ", negativetets);
-  PrintMessage (3, "illegal tets:  ", illegaltets);
-  PrintMessage (3, "bad tets:      ", badtets);
-}
+    if (!retvalues)
+      {
+	PrintMessage (1, "");
+	PrintMessage (1, "between planes:  phimin = ", (180/M_PI) * phimin,
+		      " phimax = ", (180/M_PI) *phimax);
+	PrintMessage (1, "inside planes:   phimin = ", (180/M_PI) * facephimin,
+		      " phimax = ", (180/M_PI) * facephimax);
+	PrintMessage (1, "");      
+      }
+    else
+      {
+	retvalues[0] = (180/M_PI) * facephimin;
+	retvalues[1] = (180/M_PI) * facephimax;
+	retvalues[2] = (180/M_PI) * phimin;
+	retvalues[3] = (180/M_PI) * phimax;
+      }
+    PrintMessage (3, "negative tets: ", negativetets);
+    PrintMessage (3, "illegal tets:  ", illegaltets);
+    PrintMessage (3, "bad tets:      ", badtets);
+  }
 
 
-int Mesh :: MarkIllegalElements ()
-{
-  int cnt = 0;
-  int i;
+  int Mesh :: MarkIllegalElements ()
+  {
+    int cnt = 0;
+    int i;
 
-  for (i = 1; i <= GetNE(); i++)
-    {
-      LegalTet (VolumeElement(i));
+    for (i = 1; i <= GetNE(); i++)
+      {
+	LegalTet (VolumeElement(i));
 
-      /*
-      Element & el = VolumeElement(i);
-      int leg1 = LegalTet (el);
-      el.flags.illegal_valid = 0;
-      int leg2 = LegalTet (el);
+	/*
+	  Element & el = VolumeElement(i);
+	  int leg1 = LegalTet (el);
+	  el.flags.illegal_valid = 0;
+	  int leg2 = LegalTet (el);
 
-      if (leg1 != leg2) 
-	{
+	  if (leg1 != leg2) 
+	  {
 	  cerr << "legal differs!!" << endl;
 	  (*testout) << "legal differs" << endl;
 	  (*testout) << "elnr = " << i << ", el = " << el
-		     << " leg1 = " << leg1 << ", leg2 = " << leg2 << endl;
-	}
+	  << " leg1 = " << leg1 << ", leg2 = " << leg2 << endl;
+	  }
       
-      //      el.flags.illegal = !LegalTet (el);
-      */
-      cnt += VolumeElement(i).Illegal();
-    }
-  return cnt;
-}
+	  //      el.flags.illegal = !LegalTet (el);
+	  */
+	cnt += VolumeElement(i).Illegal();
+      }
+    return cnt;
+  }
 
 #ifdef NONE
-void Mesh :: AddIdentification (int pi1, int pi2, int identnr)
-{
-  INDEX_2 pair(pi1, pi2);
-  //  pair.Sort();
-  identifiedpoints->Set (pair, identnr);
-  if (identnr > maxidentnr)
-    maxidentnr = identnr;
-  timestamp = NextTimeStamp();
-}
+  void Mesh :: AddIdentification (int pi1, int pi2, int identnr)
+  {
+    INDEX_2 pair(pi1, pi2);
+    //  pair.Sort();
+    identifiedpoints->Set (pair, identnr);
+    if (identnr > maxidentnr)
+      maxidentnr = identnr;
+    timestamp = NextTimeStamp();
+  }
 
-int Mesh :: GetIdentification (int pi1, int pi2) const
-{
-  INDEX_2 pair(pi1, pi2);
-  if (identifiedpoints->Used (pair))
-    return identifiedpoints->Get(pair);
-  else
-    return 0;
-}
+  int Mesh :: GetIdentification (int pi1, int pi2) const
+  {
+    INDEX_2 pair(pi1, pi2);
+    if (identifiedpoints->Used (pair))
+      return identifiedpoints->Get(pair);
+    else
+      return 0;
+  }
 
-int Mesh :: GetIdentificationSym (int pi1, int pi2) const
-{
-  INDEX_2 pair(pi1, pi2);
-  if (identifiedpoints->Used (pair))
-    return identifiedpoints->Get(pair);
+  int Mesh :: GetIdentificationSym (int pi1, int pi2) const
+  {
+    INDEX_2 pair(pi1, pi2);
+    if (identifiedpoints->Used (pair))
+      return identifiedpoints->Get(pair);
 
-  pair = INDEX_2 (pi2, pi1);
-  if (identifiedpoints->Used (pair))
-    return identifiedpoints->Get(pair);
+    pair = INDEX_2 (pi2, pi1);
+    if (identifiedpoints->Used (pair))
+      return identifiedpoints->Get(pair);
 
-  return 0;
-}
+    return 0;
+  }
 
 
-void Mesh :: GetIdentificationMap (int identnr, ARRAY<int> & identmap) const
-{
-  int i, j;
+  void Mesh :: GetIdentificationMap (int identnr, ARRAY<int> & identmap) const
+  {
+    int i, j;
 
-  identmap.SetSize (GetNP());
-  for (i = 1; i <= identmap.Size(); i++)
-    identmap.Elem(i) = 0;
+    identmap.SetSize (GetNP());
+    for (i = 1; i <= identmap.Size(); i++)
+      identmap.Elem(i) = 0;
 
-  for (i = 1; i <= identifiedpoints->GetNBags(); i++)
-    for (j = 1; j <= identifiedpoints->GetBagSize(i); j++)
-      {
-	INDEX_2 i2;
-	int nr;
-	identifiedpoints->GetData (i, j, i2, nr);
+    for (i = 1; i <= identifiedpoints->GetNBags(); i++)
+      for (j = 1; j <= identifiedpoints->GetBagSize(i); j++)
+	{
+	  INDEX_2 i2;
+	  int nr;
+	  identifiedpoints->GetData (i, j, i2, nr);
 	
-	if (nr == identnr)
-	  {
-	    identmap.Elem(i2.I1()) = i2.I2();
-	  }
-      }
-}
+	  if (nr == identnr)
+	    {
+	      identmap.Elem(i2.I1()) = i2.I2();
+	    }
+	}
+  }
 
 
-void Mesh :: GetIdentificationPairs (int identnr, ARRAY<INDEX_2> & identpairs) const
-{
-  int i, j;
+  void Mesh :: GetIdentificationPairs (int identnr, ARRAY<INDEX_2> & identpairs) const
+  {
+    int i, j;
 
-  identpairs.SetSize(0);
+    identpairs.SetSize(0);
 
-  for (i = 1; i <= identifiedpoints->GetNBags(); i++)
-    for (j = 1; j <= identifiedpoints->GetBagSize(i); j++)
-      {
-	INDEX_2 i2;
-	int nr;
-	identifiedpoints->GetData (i, j, i2, nr);
+    for (i = 1; i <= identifiedpoints->GetNBags(); i++)
+      for (j = 1; j <= identifiedpoints->GetBagSize(i); j++)
+	{
+	  INDEX_2 i2;
+	  int nr;
+	  identifiedpoints->GetData (i, j, i2, nr);
 	
-	if (identnr == 0 || nr == identnr)
-	  identpairs.Append (i2);
-      }
-}
+	  if (identnr == 0 || nr == identnr)
+	    identpairs.Append (i2);
+	}
+  }
 #endif
 
-void Mesh :: ComputeNVertices ()
-{
-  int i, j, nv;
-  int ne = GetNE();
-  int nse = GetNSE();
-
-  numvertices = 0;
-  for (i = 1; i <= ne; i++)
-    {
-      const Element & el = VolumeElement(i);
-      nv = el.GetNV();
-      for (j = 0; j < nv; j++)
-	if (el[j] > numvertices)
-	  numvertices = el[j];
-    }
-  for (i = 1; i <= nse; i++)
-    {
-      const Element2d & el = SurfaceElement(i);
-      nv = el.GetNV();
-      for (j = 1; j <= nv; j++)
-	if (el.PNum(j) > numvertices)
-	  numvertices = el.PNum(j);
-   } 
-
-  numvertices += 1- PointIndex::BASE;
-}
-
-int Mesh :: GetNV () const
-{
-  if (numvertices < 0)
-    return GetNP();
-  else
-    return numvertices;
-}
+  void Mesh :: ComputeNVertices ()
+  {
+    int i, j, nv;
+    int ne = GetNE();
+    int nse = GetNSE();
 
-void Mesh :: SetNP (int np)
-{
-  points.SetSize(np);
-  ptyps.SetSize(np);
-
-  int mlold = mlbetweennodes.Size();
-  mlbetweennodes.SetSize(np);
-  if (np > mlold)
-    for (int i = mlold+PointIndex::BASE; 
-	 i < np+PointIndex::BASE; i++)
+    numvertices = 0;
+    for (i = 1; i <= ne; i++)
       {
-	mlbetweennodes[i].I1() = PointIndex::BASE-1;
-	mlbetweennodes[i].I2() = PointIndex::BASE-1;
+	const Element & el = VolumeElement(i);
+	nv = el.GetNV();
+	for (j = 0; j < nv; j++)
+	  if (el[j] > numvertices)
+	    numvertices = el[j];
       }
-}
+    for (i = 1; i <= nse; i++)
+      {
+	const Element2d & el = SurfaceElement(i);
+	nv = el.GetNV();
+	for (j = 1; j <= nv; j++)
+	  if (el.PNum(j) > numvertices)
+	    numvertices = el.PNum(j);
+      } 
+
+    numvertices += 1- PointIndex::BASE;
+  }
+
+  int Mesh :: GetNV () const
+  {
+    if (numvertices < 0)
+      return GetNP();
+    else
+      return numvertices;
+  }
+
+  void Mesh :: SetNP (int np)
+  {
+    points.SetSize(np);
+    //  ptyps.SetSize(np);
+
+    int mlold = mlbetweennodes.Size();
+    mlbetweennodes.SetSize(np);
+    if (np > mlold)
+      for (int i = mlold+PointIndex::BASE; 
+	   i < np+PointIndex::BASE; i++)
+	{
+	  mlbetweennodes[i].I1() = PointIndex::BASE-1;
+	  mlbetweennodes[i].I2() = PointIndex::BASE-1;
+	}
 
+    GetIdentifications().SetMaxPointNr (np + PointIndex::BASE-1);
+  }
 
-/*
-void Mesh :: BuildConnectedNodes ()
-{
-  if (PureTetMesh())
+
+  /*
+    void Mesh :: BuildConnectedNodes ()
     {
-      connectedtonode.SetSize(0);
-      return;
+    if (PureTetMesh())
+    {
+    connectedtonode.SetSize(0);
+    return;
     }
 
 
-  int i, j, k;
-  int np = GetNP();
-  int ne = GetNE();
-  TABLE<int> conto(np);
-  for (i = 1; i <= ne; i++)
+    int i, j, k;
+    int np = GetNP();
+    int ne = GetNE();
+    TABLE<int> conto(np);
+    for (i = 1; i <= ne; i++)
     {
-      const Element & el = VolumeElement(i);
+    const Element & el = VolumeElement(i);
 
-      if (el.GetType() == PRISM)
-	{
-	  for (j = 1; j <= 6; j++)
-	    {
-	      int n1 = el.PNum (j);
-	      int n2 = el.PNum ((j+2)%6+1);
-	      //	    if (n1 != n2)
-	      {
-		int found = 0;
-		for (k = 1; k <= conto.EntrySize(n1); k++)
-		  if (conto.Get(n1, k) == n2)
-		    {
-		      found = 1;
-		      break;
-		    }
-		if (!found)
-		  conto.Add (n1, n2);
-	      }
-	    }
-	}
-      else if (el.GetType() == PYRAMID)
-	{
-	  for (j = 1; j <= 4; j++)
-	    {
-	      int n1, n2;
-	      switch (j)
-		{
-		case 1: n1 = 1; n2 = 4; break;
-		case 2: n1 = 4; n2 = 1; break;
-		case 3: n1 = 2; n2 = 3; break;
-		case 4: n1 = 3; n2 = 2; break;
-		}
+    if (el.GetType() == PRISM)
+    {
+    for (j = 1; j <= 6; j++)
+    {
+    int n1 = el.PNum (j);
+    int n2 = el.PNum ((j+2)%6+1);
+    //	    if (n1 != n2)
+    {
+    int found = 0;
+    for (k = 1; k <= conto.EntrySize(n1); k++)
+    if (conto.Get(n1, k) == n2)
+    {
+    found = 1;
+    break;
+    }
+    if (!found)
+    conto.Add (n1, n2);
+    }
+    }
+    }
+    else if (el.GetType() == PYRAMID)
+    {
+    for (j = 1; j <= 4; j++)
+    {
+    int n1, n2;
+    switch (j)
+    {
+    case 1: n1 = 1; n2 = 4; break;
+    case 2: n1 = 4; n2 = 1; break;
+    case 3: n1 = 2; n2 = 3; break;
+    case 4: n1 = 3; n2 = 2; break;
+    }
 
-	      int found = 0;
-	      for (k = 1; k <= conto.EntrySize(n1); k++)
-		if (conto.Get(n1, k) == n2)
-		  {
-		    found = 1;
-		    break;
-		  }
-	      if (!found)
-		conto.Add (n1, n2);
-	    }
-	}
+    int found = 0;
+    for (k = 1; k <= conto.EntrySize(n1); k++)
+    if (conto.Get(n1, k) == n2)
+    {
+    found = 1;
+    break;
+    }
+    if (!found)
+    conto.Add (n1, n2);
+    }
+    }
     }
   
-  connectedtonode.SetSize(np);
-  for (i = 1; i <= np; i++)
+    connectedtonode.SetSize(np);
+    for (i = 1; i <= np; i++)
     connectedtonode.Elem(i) = 0;
   
-  for (i = 1; i <= np; i++)
+    for (i = 1; i <= np; i++)
     if (connectedtonode.Elem(i) == 0)
-      {
-	connectedtonode.Elem(i) = i;
-	ConnectToNodeRec (i, i, conto);
-      }
+    {
+    connectedtonode.Elem(i) = i;
+    ConnectToNodeRec (i, i, conto);
+    }
   
 
 
-}
+    }
 
-void Mesh :: ConnectToNodeRec (int node, int tonode, 
-		       const TABLE<int> & conto)
-{
-  int i, n2;
-  //  (*testout) << "connect " << node << " to " << tonode << endl;
-  for (i = 1; i <= conto.EntrySize(node); i++)
+    void Mesh :: ConnectToNodeRec (int node, int tonode, 
+    const TABLE<int> & conto)
     {
-      n2 = conto.Get(node, i);
-      if (!connectedtonode.Get(n2))
-	{
-	  connectedtonode.Elem(n2) = tonode;
-	  ConnectToNodeRec (n2, tonode, conto);
-	}
+    int i, n2;
+    //  (*testout) << "connect " << node << " to " << tonode << endl;
+    for (i = 1; i <= conto.EntrySize(node); i++)
+    {
+    n2 = conto.Get(node, i);
+    if (!connectedtonode.Get(n2))
+    {
+    connectedtonode.Elem(n2) = tonode;
+    ConnectToNodeRec (n2, tonode, conto);
     }
-}
-*/
+    }
+    }
+  */
 
 
-bool Mesh :: PureTrigMesh (int faceindex) const
-{
-  if (!faceindex)
-    return !mparam.quad;
+  bool Mesh :: PureTrigMesh (int faceindex) const
+  {
+    if (!faceindex)
+      return !mparam.quad;
 
-  int i;
-  for (i = 1; i <= GetNSE(); i++)
-    if (SurfaceElement(i).GetIndex() == faceindex &&
-	SurfaceElement(i).GetNP() != 3)
-      return 0;
-  return 1;
-}
+    int i;
+    for (i = 1; i <= GetNSE(); i++)
+      if (SurfaceElement(i).GetIndex() == faceindex &&
+	  SurfaceElement(i).GetNP() != 3)
+	return 0;
+    return 1;
+  }
 
-bool Mesh :: PureTetMesh () const
-{
-  for (ElementIndex ei = 0; ei < GetNE(); ei++)
-    if (VolumeElement(ei).GetNP() != 4)
-      return 0;
-  return 1;
-}
+  bool Mesh :: PureTetMesh () const
+  {
+    for (ElementIndex ei = 0; ei < GetNE(); ei++)
+      if (VolumeElement(ei).GetNP() != 4)
+	return 0;
+    return 1;
+  }
 
-void Mesh :: UpdateTopology()
-{
-  topology->Update();
-  clusters->Update();
-}
+  void Mesh :: UpdateTopology()
+  {
+    topology->Update();
+    clusters->Update();
+  }
 
 
-void Mesh :: SetMaterial (int domnr, const char * mat)
-{
-  if (domnr > materials.Size())
-    {
-      int olds = materials.Size();
-      materials.SetSize (domnr);
-      for (int i = olds; i < domnr; i++)
-	materials[i] = 0;
-    }
-  materials.Elem(domnr) = new char[strlen(mat+1)];
-  strcpy (materials.Elem(domnr), mat);
-}
+  void Mesh :: SetMaterial (int domnr, const char * mat)
+  {
+    if (domnr > materials.Size())
+      {
+	int olds = materials.Size();
+	materials.SetSize (domnr);
+	for (int i = olds; i < domnr; i++)
+	  materials[i] = 0;
+      }
+    materials.Elem(domnr) = new char[strlen(mat)+1];
+    strcpy (materials.Elem(domnr), mat);
+  }
 
-const char * Mesh :: GetMaterial (int domnr) const
-{
-  if (domnr <= materials.Size())
-    return materials.Get(domnr);
-  return 0;
-}
+  const char * Mesh :: GetMaterial (int domnr) const
+  {
+    if (domnr <= materials.Size())
+      return materials.Get(domnr);
+    return 0;
+  }
 
 
 
 
 
-void Mesh :: PrintMemInfo (ostream & ost) const
-{
-  ost << "Mesh Mem:" << endl;
+  void Mesh :: PrintMemInfo (ostream & ost) const
+  {
+    ost << "Mesh Mem:" << endl;
 
-  ost << GetNP() << " Points, of size " 
-      << sizeof (Point3d) << " + " << sizeof(POINTTYPE) << " = "
-      << GetNP() * (sizeof (Point3d) + sizeof(POINTTYPE)) << endl;
+    ost << GetNP() << " Points, of size " 
+	<< sizeof (Point3d) << " + " << sizeof(POINTTYPE) << " = "
+	<< GetNP() * (sizeof (Point3d) + sizeof(POINTTYPE)) << endl;
 
-  ost << GetNSE() << " Surface elements, of size " 
-      << sizeof (Element2d) << " = " 
-      << GetNSE() * sizeof(Element2d) << endl;
+    ost << GetNSE() << " Surface elements, of size " 
+	<< sizeof (Element2d) << " = " 
+	<< GetNSE() * sizeof(Element2d) << endl;
 
-  ost << GetNE() << " Volume elements, of size " 
-      << sizeof (Element) << " = " 
-      << GetNE() * sizeof(Element) << endl;
+    ost << GetNE() << " Volume elements, of size " 
+	<< sizeof (Element) << " = " 
+	<< GetNE() * sizeof(Element) << endl;
 
-  ost << "surfs on node:";
-  surfacesonnode.PrintMemInfo (cout);
+    ost << "surfs on node:";
+    surfacesonnode.PrintMemInfo (cout);
   
-  ost << "boundaryedges: ";
-  if (boundaryedges)
-    boundaryedges->PrintMemInfo (cout);
+    ost << "boundaryedges: ";
+    if (boundaryedges)
+      boundaryedges->PrintMemInfo (cout);
 
-  ost << "surfelementht: ";
-  if (surfelementht)
-    surfelementht->PrintMemInfo (cout);
-}
+    ost << "surfelementht: ";
+    if (surfelementht)
+      surfelementht->PrintMemInfo (cout);
+  }
 }
diff --git a/Netgen/libsrc/meshing/meshclass.hpp b/Netgen/libsrc/meshing/meshclass.hpp
index 4c03a36e14..4ad0edbb9b 100644
--- a/Netgen/libsrc/meshing/meshclass.hpp
+++ b/Netgen/libsrc/meshing/meshclass.hpp
@@ -16,6 +16,9 @@
 enum resthtype { RESTRICTH_FACE, RESTRICTH_EDGE, 
 		 RESTRICTH_SURFACEELEMENT, RESTRICTH_POINT, RESTRICTH_SEGMENT };
 
+class HPRefElement;
+
+
 /// 2d/3d mesh
 class Mesh
 {
@@ -33,7 +36,7 @@ private:
   /// point coordinates
   T_POINTS points;
   /// type of point, is set in calcsurfacesofnode
-  ARRAY<POINTTYPE,PointIndex::BASE> ptyps;
+  //  ARRAY<POINTTYPE,PointIndex::BASE> ptyps;
   /// type of element, set in calcsurfacesofnode
   ARRAY<ELEMENTTYPE> eltyps;
 
@@ -52,9 +55,10 @@ private:
   /// boundary edges  (1..normal bedge, 2..segment)
   INDEX_2_CLOSED_HASHTABLE<int> * boundaryedges;
   ///
-  INDEX_2_HASHTABLE<int> * segmentht;
+  INDEX_2_CLOSED_HASHTABLE<int> * segmentht;
   ///
-  INDEX_3_HASHTABLE<int> * surfelementht;
+  INDEX_3_CLOSED_HASHTABLE<int> * surfelementht;
+
   /// faces of rest-solid
   ARRAY<Element2d> openelements;
   /// open segmenets for surface meshing  
@@ -81,7 +85,7 @@ private:
   ARRAY<char*> materials;
 
   /// Periodic surface, close surface, etc. identifications
-  Identifications ident;
+  Identifications * ident;
 
 
   /// number of vertices (if < 0, use np)
@@ -112,6 +116,10 @@ private:
 
 
 public:
+
+  // store coarse mesh before hp-refinement
+  ARRAY<HPRefElement> * hpelements;
+  Mesh * coarsemesh;
   
   
   /// number of refinement levels
@@ -130,6 +138,8 @@ public:
   ///
   ~Mesh();
 
+  Mesh & operator= (const Mesh & mesh2);
+  
   ///
   void DeleteMesh();
   
@@ -170,12 +180,17 @@ public:
   const MeshPoint & operator[] (PointIndex pi) const { return points[pi]; }
   MeshPoint & operator[] (PointIndex pi) { return points[pi]; }
 
+  /*
   POINTTYPE PointType (int i) const { return ptyps.Get(i); }
   POINTTYPE PointType (PointIndex pi) const { return ptyps[pi]; }
+  */
+  POINTTYPE PointType (int i) const { return points.Get(i).Type(); }
+  POINTTYPE PointType (PointIndex pi) const { return points[pi].Type(); }
+
 
   const T_POINTS & Points() const { return points; }
   T_POINTS & Points() { return points; }
-  ARRAY<POINTTYPE,PointIndex::BASE> & PointTypes() { return ptyps; }
+  // ARRAY<POINTTYPE,PointIndex::BASE> & PointTypes() { return ptyps; }
 
 
 
@@ -192,11 +207,11 @@ public:
   int GetNSeg () const { return segments.Size(); }
   Segment & LineSegment(int i) { return segments.Elem(i); }
   const Segment & LineSegment(int i) const { return segments.Get(i); }
-  
-  const Segment & operator[] (SegmentIndex si) const
-  { return segments[si]; }
-  Segment & operator[] (SegmentIndex si)
-  { return segments[si]; }
+
+  Segment & LineSegment(SegmentIndex si) { return segments[si]; }
+  const Segment & LineSegment(SegmentIndex si) const { return segments[si]; }
+  const Segment & operator[] (SegmentIndex si) const { return segments[si]; }
+  Segment & operator[] (SegmentIndex si) { return segments[si]; }
 
 
 
@@ -344,9 +359,11 @@ public:
   /// number of elements per radius
   void CalcLocalHFromSurfaceCurvature(double elperr);
   ///
+  void CalcLocalHFromPointDistances(void);
+  ///
   void RestrictLocalH (resthtype rht, int nr, double loch);
   ///
-  void LoadLocalMeshSize (istream & ist);
+  void LoadLocalMeshSize (const char * meshsizefilename);
   ///
   void SetGlobalH (double h);
   ///
@@ -401,15 +418,25 @@ public:
     return segmentht->Used (i2);
   }
 
+  SegmentIndex SegmentNr (PointIndex pi1, PointIndex pi2) const
+  {
+    INDEX_2 i2 (pi1, pi2);
+    i2.Sort();
+    return segmentht->Get (i2);
+  }
+
+
   /**
      Remove unused points. etc.
   */
   void Compress ();
 
   ///
-  void Save (const char * filename) const;
+  void Save (const string & filename) const;
+  ///
+  void Load (const string & filename);
   ///
-  void Load (const char * filename);
+  void Merge (const string & filename);
 
 
   ///
@@ -470,7 +497,9 @@ public:
   void BuildElementSearchTree ();
   /// gives element of point, barycentric coordinates
   int GetElementOfPoint (const Point3d & p,
-			 double * lami) const;
+			 double * lami,
+			 bool build_searchtree = 0,
+			 const int index = -1) const;
 
   /// give list of vol elements which are int the box(p1,p2)
   void GetIntersectingVolEls(const Point3d& p1, const Point3d& p2, 
@@ -529,9 +558,9 @@ public:
 #endif
 
   /// return periodic, close surface etc. identifications
-  Identifications & GetIdentifications () { return ident; }
+  Identifications & GetIdentifications () { return *ident; }
   /// return periodic, close surface etc. identifications
-  const Identifications & GetIdentifications () const { return ident; }
+  const Identifications & GetIdentifications () const { return *ident; }
 
 
 
diff --git a/Netgen/libsrc/meshing/meshfunc.cpp b/Netgen/libsrc/meshing/meshfunc.cpp
index a3459ae71f..90fe9e68c5 100644
--- a/Netgen/libsrc/meshing/meshfunc.cpp
+++ b/Netgen/libsrc/meshing/meshfunc.cpp
@@ -3,7 +3,8 @@
 
 namespace netgen
 {
-  extern const char * tetrules2[];
+  extern const char * tetrules[];
+  // extern const char * tetrules2[];
   extern const char * prismrules2[];
   extern const char * pyramidrules[];
   extern const char * pyramidrules2[];
@@ -12,38 +13,42 @@ namespace netgen
   extern double teterrpow; 
   MESHING3_RESULT MeshVolume (MeshingParameters & mp, Mesh& mesh3d)
   {
-    int i, k, oldne;
+    int i, oldne;
     PointIndex pi;
 
     int meshed;
     int cntsteps; 
 
 
-    //  PlotStatistics3d * pstat;
-    //  pstat = new TerminalPlotStatistics3d;
-
-
     ARRAY<INDEX_2> connectednodes;
 
     mesh3d.Compress();
 
     //  mesh3d.PrintMemInfo (cout);
 
+
+
+    if (mesh3d.CheckOverlappingBoundary())
+      throw NgException ("Stop meshing since boundary mesh is overlapping");
+
+
     int nonconsist = 0;
-    for (k = 1; k <= mesh3d.GetNDomains(); k++)
+    for (int k = 1; k <= mesh3d.GetNDomains(); k++)
       {
 	PrintMessage (3, "Check subdomain ", k, " / ", mesh3d.GetNDomains());
 
 	mesh3d.FindOpenElements(k);
 
-	int res = 0; // mesh3d.CheckOverlappingBoundary();
+	/*
+	bool res = mesh3d.CheckOverlappingBoundary();
 	if (res)
 	  {
 	    PrintError ("Surface is overlapping !!");
 	    nonconsist = 1;
 	  }
+	*/
 
-	res = mesh3d.CheckConsistentBoundary();
+	bool res = mesh3d.CheckConsistentBoundary();
 	if (res)
 	  {
 	    PrintError ("Surface mesh not consistent");
@@ -59,7 +64,7 @@ namespace netgen
 
     double globmaxh = mp.maxh;
 
-    for (k = 1; k <= mesh3d.GetNDomains(); k++)
+    for (int k = 1; k <= mesh3d.GetNDomains(); k++)
       {
 	if (multithread.terminate)
 	  break;
@@ -72,21 +77,6 @@ namespace netgen
 	mesh3d.CalcSurfacesOfNode();
 	mesh3d.FindOpenElements(k);
 
-	/*
-	  if (mesh3d.CheckOverlappingBoundary())
-	  {
-	  (*mycout) << "overlapping" << endl;
-	  (*testout) << "overlapping !!!" << endl;
-	  continue;
-	  }
-	  if (mesh3d.CheckConsistentBoundary())
-	  {
-	  (*mycout) << "boundary not consistent" << endl;
-	  (*testout) << "boundary not consistent !!!" << endl;
-	  continue;
-	  }
-	*/
-
 	if (!mesh3d.GetNOpenElements())
 	  continue;
 
@@ -96,32 +86,26 @@ namespace netgen
 	    if (mesh3d.HasOpenQuads())
 	      {
 		string rulefile = ngdir;
-		/*
-		  char rulefile[255];
-		  strcpy (rulefile, ngdir);
-		*/
+
 		const char ** rulep = NULL;
 		switch (qstep)
 		  {
 		  case 1:
-		    //		  strcat (rulefile, "/rules/prisms2.rls");
 		    rulefile += "/rules/prisms2.rls";
 		    rulep = prismrules2;
 		    break;
 		  case 2: // connect pyramid to triangle
-		    //		  strcat (rulefile, "/rules/pyramids2.rls");
 		    rulefile += "/rules/pyramids2.rls";
 		    rulep = pyramidrules2;
 		    break;
 		  case 3: // connect to vis-a-vis point
-		    //		  strcat (rulefile, "/rules/pyramids.rls");
 		    rulefile += "/rules/pyramids.rls";
 		    rulep = pyramidrules;
 		    break;
 		  }
 	      
-		//	      Meshing3 meshing(rulefile.c_str(), pstat);
-		Meshing3 meshing(NULL, rulep); 
+		//		Meshing3 meshing(rulefile);
+		Meshing3 meshing(rulep); 
 	      
 		MeshingParameters mpquad = mp;
 	      
@@ -168,7 +152,7 @@ namespace netgen
 
 	if (mp.delaunay && mesh3d.GetNOpenElements())
 	  {
-	    Meshing3 meshing(NULL);
+	    Meshing3 meshing((const char**)NULL);
 	 
 	    mesh3d.FindOpenElements(k);
 	  
@@ -182,21 +166,8 @@ namespace netgen
 	      meshing.AddBoundaryElement (mesh3d.OpenElement(i));
 	  
 	    oldne = mesh3d.GetNE();
-	    /*
-	      if (globflags.GetDefineFlag ("blockfill"))
-	      {
-	      if (!globflags.GetDefineFlag ("localh"))
-	      meshing.BlockFill 
-	      (mesh3d, mp.h * globflags.GetNumFlag ("relblockfillh", 1));
-	      else
-	      meshing.BlockFillLocalH (mesh3d);
-	      }
-	    */
 
-	  
-	    //	  mp.blockfill = globflags.GetDefineFlag ("blockfill");
 	    meshing.Delaunay (mesh3d, mp);
-	    //	  return MESHING3_OK;
 
 	    for (i = oldne + 1; i <= mesh3d.GetNE(); i++)
 	      mesh3d.VolumeElement(i).SetIndex (k);
@@ -212,49 +183,23 @@ namespace netgen
 	    if (multithread.terminate)
 	      break;
 
-	    //	  mesh3d.CalcSurfacesOfNode();
 	    mesh3d.FindOpenElements(k);
 	    PrintMessage (5, mesh3d.GetNOpenElements(), " open faces");
 	    cntsteps++;
+
 	    if (cntsteps > mp.maxoutersteps) 
 	      throw NgException ("Stop meshing since too many attempts");
-	    // return MESHING3_OUTERSTEPSEXCEEDED;
 
-	    /*
-	      if (mesh3d.CheckOverlappingBoundary())
-	      {
-	      (*mycout) << "overlapping" << endl;
-	      (*testout) << "overlapping !!!!" << endl;
-	      break;
-	      }
-	      if (mesh3d.CheckConsistentBoundary())
-	      {
-	      (*mycout) << "boundary not consistent" << endl;
-	      (*testout) << "boundary not consistent !!!" << endl;
-	      //	      break;
-	      }
-	    */
-
-
-	    /*
-	      char rulefile[255];
-	      strcpy (rulefile, ngdir);
-	      strcat (rulefile, "/tetra.rls");
-	    */
 	    string rulefile = ngdir + "/tetra.rls";
 	    PrintMessage (1, "start tetmeshing");
 
-	    //	  Meshing3 meshing(rulefile, pstat);
-	    Meshing3 meshing(NULL);
+	    //	  Meshing3 meshing(rulefile);
+	    Meshing3 meshing(tetrules);
       
-#ifdef MYGRAPH	  
-	    Point3d pmin, pmax;
-	    mesh3d.GetBox (pmin, pmax, k);
-	    rot.SetCenter (Center (pmin, pmax));
-#endif
 
-	    for (i = 1; i <= mesh3d.GetNP(); i++)
-	      meshing.AddPoint (mesh3d.Point(i), i);
+	    for (PointIndex pi = PointIndex::BASE; 
+		 pi < mesh3d.GetNP()+PointIndex::BASE; pi++)
+	      meshing.AddPoint (mesh3d[pi], pi);
 
 	    for (i = 1; i <= mesh3d.GetNOpenElements(); i++)
 	      meshing.AddBoundaryElement (mesh3d.OpenElement(i));
@@ -262,12 +207,12 @@ namespace netgen
 
 	    oldne = mesh3d.GetNE();
 
-	    mp.giveuptol = 15; // int(globflags.GetNumFlag ("giveuptol", 15));
+	    mp.giveuptol = 15; 
 	    mp.sloppy = 5;
 	    meshing.GenerateMesh (mesh3d, mp);
 
-	    for (i = oldne + 1; i <= mesh3d.GetNE(); i++)
-	      mesh3d.VolumeElement(i).SetIndex (k);
+	    for (ElementIndex ei = oldne; ei < mesh3d.GetNE(); ei++)
+	      mesh3d[ei].SetIndex (k);
 	  
 	  
 	    mesh3d.CalcSurfacesOfNode();
@@ -323,8 +268,6 @@ namespace netgen
 
     MeshQuality3d (mesh3d);
 
-    //  delete pstat;
-
     return MESHING3_OK;
   }  
 
@@ -646,8 +589,8 @@ namespace netgen
 
 
   MESHING3_RESULT OptimizeVolume (MeshingParameters & mp, 
-				  Mesh & mesh3d,
-				  const CSGeometry * geometry)
+				  Mesh & mesh3d)
+    //				  const CSGeometry * geometry)
   {
     int i, j;
 
diff --git a/Netgen/libsrc/meshing/meshfunc.hpp b/Netgen/libsrc/meshing/meshfunc.hpp
index 0e530ed2f0..ab2d661050 100644
--- a/Netgen/libsrc/meshing/meshfunc.hpp
+++ b/Netgen/libsrc/meshing/meshfunc.hpp
@@ -13,8 +13,7 @@
  */
 
 class Mesh;
-class MeshingParameters3;
-class CSGeometry;
+// class CSGeometry;
 
 /// Build tet-mesh
 MESHING3_RESULT MeshVolume(MeshingParameters & mp, Mesh& mesh3d);
@@ -23,8 +22,8 @@ MESHING3_RESULT MeshVolume(MeshingParameters & mp, Mesh& mesh3d);
 MESHING3_RESULT MeshMixedVolume(MeshingParameters & mp, Mesh& mesh3d);
 
 /// Optimize tet-mesh
-MESHING3_RESULT OptimizeVolume(MeshingParameters & mp, Mesh& mesh3d,
-			       const CSGeometry * geometry = NULL);
+MESHING3_RESULT OptimizeVolume(MeshingParameters & mp, Mesh& mesh3d);
+//			       const CSGeometry * geometry = NULL);
 
 void RemoveIllegalElements (Mesh & mesh3d);
 
diff --git a/Netgen/libsrc/meshing/meshing.hpp b/Netgen/libsrc/meshing/meshing.hpp
index 3a60f1f7c1..a4442867b3 100644
--- a/Netgen/libsrc/meshing/meshing.hpp
+++ b/Netgen/libsrc/meshing/meshing.hpp
@@ -10,8 +10,8 @@ namespace netgen
 {
 
   class CSGeometry;
-
-
+  
+  
 #include "msghandler.hpp"
 
 #include "meshtype.hpp"
@@ -30,15 +30,27 @@ namespace netgen
 #include "geomsearch.hpp"
 #include "adfront3.hpp"
 #include "ruler3.hpp"
+
+#ifndef SMALLLIB
+#define _INCLUDE_MORE
+#endif
+#ifdef LINUX
+#define _INCLUDE_MORE
+#endif
+
+#ifdef _INCLUDE_MORE
 #include "meshing3.hpp"
 #include "improve3.hpp"
+#endif
 #include "findip.hpp"
 
 #include "topology.hpp"
 #include "curvedelems.hpp"
 #include "clusters.hpp"
 
+#ifdef _INCLUDE_MORE
 #include "meshfunc.hpp"
+#endif
 #include "bisect.hpp"
 #include "hprefinement.hpp"
 #include "boundarylayer.hpp"
diff --git a/Netgen/libsrc/meshing/meshing2.cpp b/Netgen/libsrc/meshing/meshing2.cpp
index efcb51346e..e7643194c4 100644
--- a/Netgen/libsrc/meshing/meshing2.cpp
+++ b/Netgen/libsrc/meshing/meshing2.cpp
@@ -18,8 +18,6 @@ namespace netgen
   static int cntelem, trials, nfaces;
   static int oldnl;
   static int qualclass;
-  // static const Meshing2 * meshthis;
-
  
 
   Meshing2 :: Meshing2 (const Box3d & aboundingbox)
@@ -27,8 +25,8 @@ namespace netgen
     boundingbox = aboundingbox;
 
     LoadRules (NULL);
-    // LoadRules ("quad.rls");
-    //  LoadRules ("triangle.rls");
+    // LoadRules ("rules/quad.rls");
+    // LoadRules ("rules/triangle.rls");
 
     adfront = new AdFront2(boundingbox);
     starttime = GetTime();
@@ -115,9 +113,9 @@ namespace netgen
 				     const MultiPointGeomInfo & geominf,
 				     Point2d & plainpoint, double h, int & zone)
   {
-    Vec3d p1p;
+    Vec3d p1p (globp1, locpoint);
 
-    p1p = locpoint - globp1;
+    //    p1p = locpoint - globp1;
     p1p /= h;
     plainpoint.X() = p1p * ex;
     plainpoint.Y() = p1p * ey;
@@ -227,7 +225,6 @@ namespace netgen
 
 
     testmode = 0;
-    //  meshthis = this;
 
     StartMesh();
 
@@ -293,7 +290,7 @@ namespace netgen
     adfront ->SetStartFront ();
 
 
-    int plotnexttrial = 0;
+    int plotnexttrial = 999;
     //  starttime = GetTime();
     while (!adfront ->Empty()) //  && !multithread.terminate)
       {
@@ -327,7 +324,7 @@ namespace netgen
 			  " trials = ", trials,
 			  " elements = ", mesh.GetNSE(),
 			  " els/sec = ",
-			  (mesh.GetNSE() / (GetTime() - starttime + 1)));
+			  (mesh.GetNSE() / (GetTime() - starttime + 0.0001)));
 	    plotnexttrial += 1000;
 	  }
 
@@ -354,7 +351,7 @@ namespace netgen
 	  }
 
 
-	baselineindex = adfront -> SelectBaseLine (p1, p2, blgeominfo1, blgeominfo2);
+	baselineindex = adfront -> SelectBaseLine (p1, p2, blgeominfo1, blgeominfo2, qualclass);
 
 	// cout << "baseline = " << baselineindex << ", p1, p2 = " << p1 << ", " << p2 << endl;
 
@@ -372,24 +369,26 @@ namespace netgen
 
 	h = hshould;
 
-      
-	qualclass =
-	  adfront ->GetLocals (baselineindex, locpoints, mpgeominfo, loclines, 
-			       pindex, lindex, 5 * max2(his, hshould));
+	double hinner = (3 + qualclass) * max2 (his, hshould);
 
+	adfront ->GetLocals (baselineindex, locpoints, mpgeominfo, loclines, 
+			     pindex, lindex, 2*hinner);
+	
 
 	if (qualclass > 200)
 	  {
 	    PrintMessage (3, "give up with qualclass ", qualclass);
 	    PrintMessage (3, "number of frontlines = ", adfront->GetNFL());
+	    // throw NgException ("Give up 2d meshing");
 	    break;
 	  }
 
+	/*
 	if (found && qualclass > 60)
 	  {
 	    found = 0;
 	  }
-
+	*/
 	//      morerisc = ((qualclass > 20) && (qualclass % 2 == 1));
 	//      morerisc = 1;
 	morerisc = 0;
@@ -416,6 +415,9 @@ namespace netgen
 	    cout << "set debugflag" << endl;
 	  }
 	
+	if (debugparam.haltlargequalclass && qualclass > 50)
+	  debugflag = 1;
+
 
 	// problem recognition !
 	if (found && 
@@ -434,6 +436,9 @@ namespace netgen
 	    oldnp = locpoints.Size();
 	    oldnl = loclines.Size();
 	  
+	    if (debugflag)
+	      (*testout) << "define new transformation" << endl;
+
 	    DefineTransformation (p1, p2, blgeominfo1, blgeominfo2);
 	  
 	    plainpoints.SetSize (locpoints.Size());
@@ -441,13 +446,18 @@ namespace netgen
 
 	    // (*testout) << endl;
 
+	    //	    (*testout) << "3d->2d transformation" << endl;
+
 	    for (i = 1; i <= locpoints.Size(); i++)
 	      {
 		// (*testout) << "pindex(i) = " << pindex[i-1] << endl;
 		TransformToPlain (locpoints.Get(i), 
 				  mpgeominfo.Get(i),
 				  plainpoints.Elem(i), h, plainzones.Elem(i));
+		//		(*testout) << mpgeominfo.Get(i).GetPGI(1).u << " " << mpgeominfo.Get(i).GetPGI(1).v << " ";
+		//		(*testout) << plainpoints.Get(i).X() << " " << plainpoints.Get(i).Y() << endl;
 	      }
+	    //	    (*testout) << endl << endl << endl;
 
 	    p12d = plainpoints.Get(1);
 	    p22d = plainpoints.Get(2);
@@ -581,6 +591,9 @@ namespace netgen
 		  }
 		if (pindex.Elem(i) == 0)
 		  legalpoints.Elem(i) = 0;
+
+		if (plainpoints.Elem(i).Y() < 0)
+		  legalpoints.Elem(i) = 0;
 	      }
 	    /*
 	      for (i = 3; i <= plainpoints.Size(); i++)
@@ -666,12 +679,12 @@ namespace netgen
 	  }
 	*/
 
-
 	if (found)
 	  {
 	    rulenr = ApplyRules (plainpoints, legalpoints, maxlegalpoint,
 				 loclines, maxlegalline, locelements,
 				 dellines, qualclass);
+	    //	    (*testout) << "Rule Nr = " << rulenr << endl;
 	    if (!rulenr)
 	      {
 		found = 0;
@@ -683,6 +696,7 @@ namespace netgen
 	for (i = 1; i <= locelements.Size() && found; i++)
 	  {
 	    const Element2d & el = locelements.Get(i);
+
 	    for (j = 1; j <= el.GetNP(); j++)
 	      if (el.PNum(j) <= oldnp && !pindex.Get(el.PNum(j)))
 		{
@@ -781,6 +795,11 @@ namespace netgen
 		  minh = eh;
 	      }
 
+	    for (i = 1; i <= locelements.Size(); i++)
+	      for (j = 1; j <= locelements.Get(i).GetNP(); j++)
+		if (Dist2 (locpoints.Get(locelements.Get(i).PNum(j)), pmid) > hinner*hinner)
+		  found = 0;
+
 	    //	  cout << "violate = " << newedgemaxh / minh << endl;
 	    static double maxviolate = 0;
 	    if (newedgemaxh / minh > maxviolate)
@@ -877,6 +896,7 @@ namespace netgen
 
 	if (found && mparam.checkoverlap)
 	  {
+	    // cout << "checkoverlap" << endl;
 	    // test for overlaps
 	  
 	    Point3d hullmin(1e10, 1e10, 1e10);
@@ -1265,7 +1285,7 @@ namespace netgen
 	    for (i = 1; i <= dellines.Size(); i++)
 	      adfront -> DeleteLine (lindex.Get(dellines.Get(i)));
 	      
-	    //	  rname = rules[found]->Name();
+	    //	  rname = rules.Get(rulenr)->Name();
 #ifdef MYGRAPH
 	    if (silentflag<3) 
 	      {
@@ -1286,7 +1306,7 @@ namespace netgen
 	  
 	    if ( debugparam.haltsuccess || debugflag )
 	      {
-		cout << "success" << endl;
+		cout << "success of rule" << rules.Get(rulenr)->Name() << endl;
 		multithread.drawing = 1;
 		multithread.testmode = 1;
 		multithread.pause = 1;
@@ -1302,6 +1322,31 @@ namespace netgen
 		  }
 		*/
 
+		(*testout) << "success of rule" << rules.Get(rulenr)->Name() << endl;
+		(*testout) << "trials = " << trials << endl;
+
+		(*testout) << "old number of lines = " << oldnl << endl;
+		for (i = 1; i <= loclines.Size(); i++)
+		  {
+		    (*testout) << "line ";
+		    for (j = 1; j <= 2; j++)
+		      {
+			int hi = 0;
+			if (loclines.Get(i).I(j) >= 1 &&
+			    loclines.Get(i).I(j) <= pindex.Size())
+			  hi = adfront->GetGlobalIndex (pindex.Get(loclines.Get(i).I(j)));
+
+			(*testout) << hi << " ";
+		      }
+		    (*testout) << " : " 
+			       << plainpoints.Get(loclines.Get(i).I1()) << " - "
+			       << plainpoints.Get(loclines.Get(i).I2()) << " 3d: "
+			       << locpoints.Get(loclines.Get(i).I1()) << " - "
+			       << locpoints.Get(loclines.Get(i).I2()) 
+			       << endl;
+		  }
+
+
 
 		glrender(1);
 	      }
@@ -1314,6 +1359,10 @@ namespace netgen
 	      {
 		cout << "Problem with seg " << gpi1 << " - " << gpi2
 		     << ", class = " << qualclass << endl;
+
+		(*testout) << "Problem with seg " << gpi1 << " - " << gpi2
+			   << ", class = " << qualclass << endl;
+
 		multithread.drawing = 1;
 		multithread.testmode = 1;
 		multithread.pause = 1;
@@ -1456,12 +1505,177 @@ namespace netgen
   {
     int i, j, k;
 
+    if (loclines.Size() != changeval)
+      {
+	center = Point<3>(0,0,-5);
+	rad = 0.1;
+  
+	CalcTransformationMatrices();
+	changeval = loclines.Size();
+      }
+
+  glClearColor(backcolor, backcolor, backcolor, 1.0);
+  glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
+  SetLight();
+
+  //  glEnable (GL_COLOR_MATERIAL);
+
+  //  glDisable (GL_SHADING);
+  //  glColor3f (0.0f, 1.0f, 1.0f);
+  //  glLineWidth (1.0f);
+  //  glShadeModel (GL_SMOOTH);
+
+  //  glCallList (linelists.Get(1));
+
+  //  SetLight();
+
+  glPushMatrix();
+  glMultMatrixf (transformationmat);
+
+  glShadeModel (GL_SMOOTH);
+  glDisable (GL_COLOR_MATERIAL);
+  glPolygonMode (GL_FRONT_AND_BACK, GL_FILL);
+
+  glEnable (GL_BLEND);
+  glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
+
+  //  glEnable (GL_LIGHTING);
+
+  double shine = vispar.shininess;
+  double transp = vispar.transp;
+
+  glMaterialf (GL_FRONT_AND_BACK, GL_SHININESS, shine);
+  glLogicOp (GL_COPY);
+
+
+
+  /*
+
+  float mat_col[] = { 0.2, 0.2, 0.8, 1 };
+  glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col);
+
+  glPolygonOffset (1, 1);
+  glEnable (GL_POLYGON_OFFSET_FILL);
+
+    float mat_colbl[] = { 0.8, 0.2, 0.2, 1 };
+    float mat_cololdl[] = { 0.2, 0.8, 0.2, 1 };
+    float mat_colnewl[] = { 0.8, 0.8, 0.2, 1 };
+
+
+    glPolygonMode (GL_FRONT_AND_BACK, GL_FILL);
+    glPolygonOffset (1, -1);
+    glLineWidth (3);
+
+    for (i = 1; i <= loclines.Size(); i++)
+      {
+	if (i == 1)
+	  {
+	    glEnable (GL_POLYGON_OFFSET_FILL);
+	    glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_colbl);
+	  }
+	else if (i <= oldnl) 
+	  glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_cololdl);
+	else 
+	  glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_colnewl);
+
+	int pi1 = loclines.Get(i).I1();
+	int pi2 = loclines.Get(i).I2();
+
+	if (pi1 >= 1 && pi2 >= 1)
+	  {
+	    Point3d p1 = locpoints.Get(pi1);
+	    Point3d p2 = locpoints.Get(pi2);
+	  
+	    glBegin (GL_LINES);
+	    glVertex3f (p1.X(), p1.Y(), p1.Z());
+	    glVertex3f (p2.X(), p2.Y(), p2.Z());
+	    glEnd();
+	  }
+
+	glDisable (GL_POLYGON_OFFSET_FILL);
+      }
+  
+
+    glLineWidth (1);
+
+
+    glPointSize (5);
+    float mat_colp[] = { 1, 0, 0, 1 };
+    glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_colp);
+    glBegin (GL_POINTS);
+    for (i = 1; i <= locpoints.Size(); i++)
+      {
+	Point3d p = locpoints.Get(i);
+	glVertex3f (p.X(), p.Y(), p.Z());
+      }
+    glEnd();
+
+
+    glPopMatrix();
+  */
+
+    float mat_colp[] = { 1, 0, 0, 1 };
+
+    float mat_col2d1[] = { 1, 0.5, 0.5, 1 };
+    float mat_col2d[] = { 1, 1, 1, 1 };
+    glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col2d);
+  
+    double scalex = 0.1, scaley = 0.1;
+
+    glBegin (GL_LINES);
+    for (i = 1; i <= loclines.Size(); i++)
+      {
+	glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col2d);
+	if (i == 1)
+	  glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col2d1);
+
+	int pi1 = loclines.Get(i).I1();
+	int pi2 = loclines.Get(i).I2();
+
+	if (pi1 >= 1 && pi2 >= 1)
+	  {
+	    Point2d p1 = plainpoints.Get(pi1);
+	    Point2d p2 = plainpoints.Get(pi2);
+	  
+	    glBegin (GL_LINES);
+	    glVertex3f (scalex * p1.X(), scaley * p1.Y(), -5);
+	    glVertex3f (scalex * p2.X(), scaley * p2.Y(), -5);
+	    glEnd();
+	  }
+      }
+    glEnd ();
+
+
+    glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_colp);
+    glBegin (GL_POINTS);
+    for (i = 1; i <= plainpoints.Size(); i++)
+      {
+	Point2d p = plainpoints.Get(i);
+	glVertex3f (scalex * p.X(), scaley * p.Y(), -5);
+      }
+    glEnd();
+
+
+
+
+
+
+  glDisable (GL_POLYGON_OFFSET_FILL);
+ 
+  glPopMatrix();
+  DrawCoordinateCross ();
+  DrawNetgenLogo ();
+  glFinish();  
+
+  /*
+    glDisable (GL_POLYGON_OFFSET_FILL);
+
     //  cout << "draw surfacemeshing" << endl;
-    /*
-      if (changeval != stlgeometry->GetNT())
-      BuildScene();
-      changeval = stlgeometry->GetNT();
-    */
+    //
+    //  if (changeval != stlgeometry->GetNT())
+    //      BuildScene();
+    //      changeval = stlgeometry->GetNT();
+    
 
     glClearColor(backcolor, backcolor, backcolor, 1.0);
     glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
@@ -1472,7 +1686,6 @@ namespace netgen
     glLoadMatrixf (transmat);
     glMultMatrixf (rotmat);
 
-
     glShadeModel (GL_SMOOTH);
     glDisable (GL_COLOR_MATERIAL);
     glPolygonMode (GL_FRONT_AND_BACK, GL_FILL);
@@ -1500,29 +1713,29 @@ namespace netgen
     glColor3f (1.0f, 1.0f, 1.0f);
 
     glEnable (GL_NORMALIZE);
-    /*
-      glBegin (GL_TRIANGLES);
-      for (j = 1; j <= stlgeometry -> GetNT(); j++)
-      {
-      glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col);
-      if (j == geomtrig)
-      glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_colrt);
+    
+    //  glBegin (GL_TRIANGLES);
+    //      for (j = 1; j <= stlgeometry -> GetNT(); j++)
+    //      {
+    //      glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col);
+    //      if (j == geomtrig)
+    //      glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_colrt);
 	
 
-      const STLReadTriangle & tria = stlgeometry -> GetReadTriangle(j);
-      glNormal3f (tria.normal.X(),
-      tria.normal.Y(),
-      tria.normal.Z());
+    //      const STLReadTriangle & tria = stlgeometry -> GetReadTriangle(j);
+    //      glNormal3f (tria.normal.X(),
+    //      tria.normal.Y(),
+    //      tria.normal.Z());
 		  
-      for (k = 0; k < 3; k++)
-      {
-      glVertex3f (tria.pts[k].X(),
-      tria.pts[k].Y(),
-      tria.pts[k].Z());
-      }
-      }    
-      glEnd ();
-    */
+    //      for (k = 0; k < 3; k++)
+    //      {
+    //      glVertex3f (tria.pts[k].X(),
+    //      tria.pts[k].Y(),
+    //      tria.pts[k].Z());
+    //      }
+    //      }    
+    //      glEnd ();
+    
 
 
     glDisable (GL_POLYGON_OFFSET_FILL);
@@ -1624,6 +1837,7 @@ namespace netgen
     glEnd();
 
     glFinish();  
+*/
   }
 
 
diff --git a/Netgen/libsrc/meshing/meshing3.cpp b/Netgen/libsrc/meshing/meshing3.cpp
index 46b89b28cd..c349b85e79 100644
--- a/Netgen/libsrc/meshing/meshing3.cpp
+++ b/Netgen/libsrc/meshing/meshing3.cpp
@@ -11,27 +11,6 @@ double minwithoutother;
 
 
 
-
-
-/*
-MeshingParameters3 ::   MeshingParameters3()
-{
-  h = 1;
-  giveuptol = 10;
-  maxoutersteps = 5;
-  baseelnp = 0;
-  starshapeclass = 5;
-  blockfill = 0;
-  sloppy = 1;
-  optsteps = 2;
-  optstr = "cmsmdm";
-}
-*/
-
-
-
-
-  
 MeshingStat3d :: MeshingStat3d ()
 {
   cntsucc = cnttrials = cntelem = qualclass = 0;
@@ -40,16 +19,11 @@ MeshingStat3d :: MeshingStat3d ()
 }  
   
 
- 
-
-
-Meshing3 :: Meshing3 (const char * rulefilename) //, const PlotStatistics3d * aplotstat)
-  //  : plotstat(aplotstat)
+Meshing3 :: Meshing3 (const string & rulefilename) 
 {
-  int i;
   tolfak = 1;
 
-  LoadRules (rulefilename, NULL);
+  LoadRules (rulefilename.c_str(), NULL);
   adfront = new AdFront3;
 
   problems.SetSize (rules.Size());
@@ -57,7 +31,7 @@ Meshing3 :: Meshing3 (const char * rulefilename) //, const PlotStatistics3d * ap
   canuse.SetSize (rules.Size());
   ruleused.SetSize (rules.Size());
 
-  for (i = 1; i <= rules.Size(); i++)
+  for (int i = 1; i <= rules.Size(); i++)
     {
       problems.Elem(i) = new char[255];
       foundmap.Elem(i) = 0;
@@ -66,9 +40,8 @@ Meshing3 :: Meshing3 (const char * rulefilename) //, const PlotStatistics3d * ap
     }
 }
 
-Meshing3 :: Meshing3 (const char * rulefilename, const char ** rulep)
-  //		      const PlotStatistics3d * aplotstat)
-  //  : plotstat(aplotstat)
+
+Meshing3 :: Meshing3 (const char ** rulep)
 {
   tolfak = 1;
 
@@ -94,35 +67,12 @@ Meshing3 :: ~Meshing3 ()
   delete adfront;
   for (int i = 0; i < rules.Size(); i++)
     {
-      delete problems[i];
+      delete [] problems[i];
       delete rules[i];
     }
 }
 
 
-#ifdef ABC
-TerminalPlotStatistics3d :: TerminalPlotStatistics3d()
-{
-  oldne = -1;
-}
-
-void TerminalPlotStatistics3d :: Plot (const MeshingStat3d & stat) const
-{
-  if (stat.cntelem == oldne)
-    return;
-  
-  ((int&)oldne) = stat.cntelem;
-
-  PrintMessageCR (5, "El: ", stat.cntelem,
-		//	    << " trials: " << stat.cnttrials
-		" faces: ", stat.nff,
-		" vol = ", float(100 * stat.vol / stat.vol0));
-  
-  multithread.percent = 100 -  100.0 * stat.vol / stat.vol0;
-}
-
-#endif
-
 
 static double CalcLocH (const ARRAY<Point3d> & locpoints,
 			const ARRAY<Element2d> & locfaces,
@@ -190,7 +140,7 @@ static double CalcLocH (const ARRAY<Point3d> & locpoints,
 }
 
 
-int Meshing3 :: AddPoint (const Point3d & p, INDEX globind)
+PointIndex Meshing3 :: AddPoint (const Point3d & p, PointIndex globind)
 {
   return adfront -> AddPoint (p, globind);  
 }  
@@ -209,8 +159,8 @@ MESHING3_RESULT Meshing3 ::
 GenerateMesh (Mesh & mesh, const MeshingParameters & mp)
 {
   ARRAY<Point3d> locpoints;      // local points
-  ARRAY<Element2d> locfaces;    // local faces
-  ARRAY<PointIndex> pindex;           // mapping from local to front point numbering
+  ARRAY<Element2d> locfaces;     // local faces
+  ARRAY<PointIndex> pindex;      // mapping from local to front point numbering
   ARRAY<int> allowpoint;         // point is allowd ?
   ARRAY<INDEX> findex;           // mapping from local to front face numbering
   INDEX_2_HASHTABLE<int> connectedpairs(100);  // connecgted pairs for prism meshing
@@ -272,10 +222,7 @@ GenerateMesh (Mesh & mesh, const MeshingParameters & mp)
     {
       if (multithread.terminate)
 	throw NgException ("Meshing stopped");
-      /*
-      if (multithread.terminate)
-	break;
-      */
+
       if (giveup)
 	break;
 
@@ -295,7 +242,7 @@ GenerateMesh (Mesh & mesh, const MeshingParameters & mp)
       pindex.SetSize(0);
       findex.SetSize(0);
 
-      INDEX_2_HASHTABLE<int> connectedpairs(100);  // connecgted pairs for prism meshing
+      INDEX_2_HASHTABLE<int> connectedpairs(100);  // connected pairs for prism meshing
       
       // select base-element (will be locface[1])
       // and get local environment of radius (safety * h)
@@ -313,6 +260,9 @@ GenerateMesh (Mesh & mesh, const MeshingParameters & mp)
       const Point3d & p2 = adfront->GetPoint (bel.PNum(2));
       const Point3d & p3 = adfront->GetPoint (bel.PNum(3));
 
+      (*testout) << endl << "base = " << bel << endl;
+
+
       Point3d pmid = Center (p1, p2, p3);
 
       double his = (Dist (p1, p2) + Dist(p1, p3) + Dist(p2, p3)) / 3;
@@ -336,9 +286,9 @@ GenerateMesh (Mesh & mesh, const MeshingParameters & mp)
 			      locfacesplit);
 
 
-      int pi1 = pindex.Get(locfaces.Elem(1).PNum(1));
-      int pi2 = pindex.Get(locfaces.Elem(1).PNum(2));
-      int pi3 = pindex.Get(locfaces.Elem(1).PNum(3));
+      int pi1 = pindex.Get(locfaces[0].PNum(1));
+      int pi2 = pindex.Get(locfaces[0].PNum(2));
+      int pi3 = pindex.Get(locfaces[0].PNum(3));
 
       /*      
       (*testout) << "baseel = " << baseelem << ", ind = " << findex.Get(1) << endl;
@@ -346,23 +296,17 @@ GenerateMesh (Mesh & mesh, const MeshingParameters & mp)
       */
 
       loktestmode = 0;
-      /*
-      // 1085, 1084, 1491 
-      if ( (pi1 == 1085 || pi1 == 1084 || pi1 == 1491) &&
-	   (pi2 == 1085 || pi2 == 1084 || pi2 == 1491) &&
-	   (pi3 == 1085 || pi3 == 1084 || pi3 == 1491) )
-	//      if (findex.Get(1) == 7551)
+      //      testmode = loktestmode;
+      // loktestmode = testmode =  (adfront->GetFace (baseelem).GetNP() == 4) && (rules.Size() == 5);
+
+      if (testmode)
 	{
-	  (*testout) << "baseel = " << findex.Get(1) << ", points: ";
-	  for (i = 1; i <= pindex.Size(); i++)
-	    (*testout) << pindex.Get(i) << " ";
-	  (*testout) << endl;
-	  loktestmode = 1;
+	  (*testout) << "baseelem = " << baseelem << " qualclass = " << stat.qualclass << endl;
+	  (*testout) << "locpoints = " << endl << locpoints << endl;
+	  (*testout) << "connected = " << endl << connectedpairs << endl;
 	}
-      */
-      //      testmode = loktestmode;
 
-      //      cout << "baseelem = " << baseelem << " qualclass = " << stat.qualclass << endl;
+
 
       // loch = CalcLocH (locpoints, locfaces, h);
       
@@ -385,6 +329,8 @@ GenerateMesh (Mesh & mesh, const MeshingParameters & mp)
 	for (i = 1; i <= allowpoint.Size(); i++)	
 	  allowpoint.Elem(i) = 1;
 
+      
+
 
       
       if (stat.qualclass >= mp.starshapeclass)   
@@ -488,7 +434,18 @@ GenerateMesh (Mesh & mesh, const MeshingParameters & mp)
 	    }
 
 	  trans.ToPlain (locpoints, plainpoints);
+
+
+	  for (i = 1; i <= allowpoint.Size(); i++)
+	    {
+	      if (plainpoints.Get(i).Z() > 0)
+		allowpoint.Elem(i) = 0;
+	    }
+
 	  
+	  if (loktestmode)
+	    (*testout) << "plainpoints = " << endl << plainpoints << endl;
+
 	  stat.cnttrials++;
 
 
diff --git a/Netgen/libsrc/meshing/meshing3.hpp b/Netgen/libsrc/meshing/meshing3.hpp
index e6829554fa..45f44751e3 100644
--- a/Netgen/libsrc/meshing/meshing3.hpp
+++ b/Netgen/libsrc/meshing/meshing3.hpp
@@ -4,47 +4,6 @@
 
 
 
-// class PlotStatistics3d;
-
-
-/*
-///
-class MeshingParameters3
-{
-public:
-  /// global h
-  double h;
-
-  /// give up quality class
-  int giveuptol;
-
-  /// maximal outer steps
-  int maxoutersteps;   
-
-  /// if non-zero, baseelement must have baseelnp points
-  int baseelnp;        
-
-  /// class starting star-shape filling
-  int starshapeclass;
-
-  /// call blockfill function
-  int blockfill;
-
-  /// quality tolerances are handled less careful
-  double sloppy;
-
-  /// number of optimizationsteps
-  int optsteps;
-
-  ///
-  char * optstr;
-
-  ///
-  MeshingParameters3();
-};
-*/
-
-
 enum MESHING3_RESULT
 {
   MESHING3_OK = 0,
@@ -71,9 +30,9 @@ class Meshing3
   double tolfak;
 public:
   /// 
-  Meshing3 (const char * rulefilename); 
+  Meshing3 (const string & rulefilename); 
   /// 
-  Meshing3 (const char * rulefilename, const char ** rulep);
+  Meshing3 (const char ** rulep);
   ///
   virtual ~Meshing3 ();
   
@@ -92,7 +51,7 @@ public:
 		  float & retminerr);
   
   ///
-  int AddPoint (const Point3d & p, INDEX globind);
+  PointIndex AddPoint (const Point3d & p, PointIndex globind);
   ///
   void AddBoundaryElement (const Element2d & elem);
   ///
@@ -141,29 +100,6 @@ public:
 };
 
 
-/*
-///
-class PlotStatistics3d
-{
-public:
-  ///
-  PlotStatistics3d () { }
-  ///
-  virtual void Plot (const MeshingStat3d & stat) const = 0;
-};
-
-
-///
-class TerminalPlotStatistics3d : public PlotStatistics3d
-{
-  int oldne;
-public:
-  ///
-  TerminalPlotStatistics3d ();
-  ///
-  virtual void Plot (const MeshingStat3d & stat) const;
-};
-*/
 
 
 
diff --git a/Netgen/libsrc/meshing/meshtool.cpp b/Netgen/libsrc/meshing/meshtool.cpp
index 127d047820..89bb2b16dd 100644
--- a/Netgen/libsrc/meshing/meshtool.cpp
+++ b/Netgen/libsrc/meshing/meshtool.cpp
@@ -7,113 +7,113 @@
 namespace netgen
 {
 
-int CheckSurfaceMesh (const Mesh & mesh)
-{
-  PrintMessage (3, "Check Surface mesh");
+  int CheckSurfaceMesh (const Mesh & mesh)
+  {
+    PrintMessage (3, "Check Surface mesh");
 
-  int nf = mesh.GetNSE();
-  INDEX_2_HASHTABLE<int> edges(nf+2);
-  int i, j;
-  INDEX_2 i2;
-  int cnt1 = 0, cnt2 = 0;
+    int nf = mesh.GetNSE();
+    INDEX_2_HASHTABLE<int> edges(nf+2);
+    int i, j;
+    INDEX_2 i2;
+    int cnt1 = 0, cnt2 = 0;
+
+    for (i = 1; i <= nf; i++)
+      for (j = 1; j <= 3; j++)
+	{
+	  i2.I1() = mesh.SurfaceElement(i).PNumMod(j);
+	  i2.I2() = mesh.SurfaceElement(i).PNumMod(j+1);
+	  if (edges.Used(i2))
+	    {
+	      int hi;
+	      hi = edges.Get(i2);
+	      if (hi != 1) 
+		PrintSysError ("CheckSurfaceMesh, hi = ", hi);
+	      edges.Set(i2, 2);
+	      cnt2++;
+	    }
+	  else
+	    {
+	      Swap (i2.I1(), i2.I2());
+	      edges.Set(i2, 1);
+	      cnt1++;
+	    }
+	}
   
-  std::cout<<"MeshTool"<<std::endl;
 
-  for (i = 1; i <= nf; i++)
-    for (j = 1; j <= 3; j++)
+    if (cnt1 != cnt2)
       {
-	i2.I1() = mesh.SurfaceElement(i).PNumMod(j);
- 	i2.I2() = mesh.SurfaceElement(i).PNumMod(j+1);
-        if (edges.Used(i2))
-          {
-            int hi;
-	    hi = edges.Get(i2);
-	    if (hi != 1) 
-	      PrintSysError ("CheckSurfaceMesh, hi = ", hi);
-            edges.Set(i2, 2);
-	    cnt2++;
-          }
-        else
-          {
-	    Swap (i2.I1(), i2.I2());
-            edges.Set(i2, 1);
-	    cnt1++;
-          }
+	PrintUserError ("Surface mesh not consistent");
+	//      MyBeep(2);
+	//      (*mycout) << "cnt1 = " << cnt1 << " cnt2 = " << cnt2 << endl;
+	return 0;
       }
-  
+    return 1;
+  }
 
-  if (cnt1 != cnt2)
-    {
-      PrintUserError ("Surface mesh not consistent");
-      //      MyBeep(2);
-      //      (*mycout) << "cnt1 = " << cnt1 << " cnt2 = " << cnt2 << endl;
-      return 0;
-    }
-  return 1;
-}
 
 
+  int CheckSurfaceMesh2 (const Mesh & mesh)
+  {
+    int i, j, k;
+    const Point3d *tri1[3], *tri2[3];
 
-int CheckSurfaceMesh2 (const Mesh & mesh)
-{
-  int i, j, k;
-  const Point3d *tri1[3], *tri2[3];
+    for (i = 1; i <= mesh.GetNOpenElements(); i++)
+      {
+	PrintDot ();
+	for (j = 1; j < i; j++)
+	  {
+	    for (k = 1; k <= 3; k++)
+	      {
+		tri1[k-1] = &mesh.Point (mesh.OpenElement(i).PNum(k));
+		tri2[k-1] = &mesh.Point (mesh.OpenElement(j).PNum(k));
+	      }
+	    if (IntersectTriangleTriangle (&tri1[0], &tri2[0]))
+	      {
+		PrintSysError ("Surface elements are intersecting");
+		(*testout) << "Intersecting: " << endl;
+		for (k = 0; k <= 2; k++)
+		  (*testout) << *tri1[k] << "   ";
+		(*testout) << endl;
+		for (k = 0; k <= 2; k++)
+		  (*testout) << *tri2[k] << "   ";
+		(*testout) << endl;
+	      }
+
+	  }
+      }
+    return 0;
+  }
 
-  for (i = 1; i <= mesh.GetNOpenElements(); i++)
-    {
-      PrintDot ();
-      for (j = 1; j < i; j++)
-	{
-	  for (k = 1; k <= 3; k++)
-	    {
-	      tri1[k-1] = &mesh.Point (mesh.OpenElement(i).PNum(k));
-	      tri2[k-1] = &mesh.Point (mesh.OpenElement(j).PNum(k));
-	    }
-	  if (IntersectTriangleTriangle (&tri1[0], &tri2[0]))
-	    {
-	      PrintSysError ("Surface elements are intersecting");
-	      (*testout) << "Intersecting: " << endl;
-	      for (k = 0; k <= 2; k++)
-		(*testout) << *tri1[k] << "   ";
-	      (*testout) << endl;
-	      for (k = 0; k <= 2; k++)
-		(*testout) << *tri2[k] << "   ";
-	      (*testout) << endl;
-	    }
 
-	}
-    }
-  return 0;
-}
 
 
 
+  static double TriangleQualityInst (const Point3d & p1, const Point3d & p2,
+				     const Point3d & p3)
+  {
+    // quality 0 (worst) .. 1 (optimal)
 
+    Vec3d v1, v2, v3;
+    double s1, s2, s3;
+    double an1, an2, an3;
 
-static double TriangleQualityInst (const Point3d & p1, const Point3d & p2,
-				   const Point3d & p3)
-{
-  // quality 0 (worst) .. 1 (optimal)
+    v1 = p2 - p1;
+    v2 = p3 - p1;
+    v3 = p3 - p2;
 
-  Vec3d v1, v2, v3;
-  double s1, s2, s3;
-  double an1, an2, an3;
+    an1 = Angle (v1, v2);
+    v1 *= -1;
+    an2 = Angle (v1, v3);
+    an3 = Angle (v2, v3);
 
-  v1 = p2 - p1;
-  v2 = p3 - p1;
-  v3 = p3 - p2;
+    s1 = sin (an1/2);
+    s2 = sin (an2/2);
+    s3 = sin (an3/2);
 
-  an1 = Angle (v1, v2);
-  v1 *= -1;
-  an2 = Angle (v1, v3);
-  an3 = Angle (v2, v3);
+    return 8 * s1 * s2 * s3;
+  }
 
-  s1 = sin (an1/2);
-  s2 = sin (an2/2);
-  s3 = sin (an3/2);
 
-  return 8 * s1 * s2 * s3;
-}
 
 
 
@@ -126,118 +126,227 @@ static double TriangleQualityInst (const Point3d & p1, const Point3d & p2,
 
 
 
+  void MeshQuality2d (const Mesh & mesh)
+  {
+    int ncl = 20, cl;
+    ARRAY<INDEX> incl(ncl);
+    INDEX i;
+    SurfaceElementIndex sei;
+    double qual;
 
+    incl = 0;
 
-void MeshQuality2d (const Mesh & mesh)
-{
-  int ncl = 20, cl;
-  ARRAY<INDEX> incl(ncl);
-  INDEX i;
-  SurfaceElementIndex sei;
-  double qual;
+    for (sei = 0; sei < mesh.GetNSE(); sei++)
+      {
+	qual = TriangleQualityInst (mesh[mesh[sei][0]],
+				    mesh[mesh[sei][1]],
+				    mesh[mesh[sei][2]]);
 
-  incl = 0;
+	cl = int ( (ncl-1e-3) * qual ) + 1;
+	incl.Elem(cl)++;
+      }
 
-  for (sei = 0; sei < mesh.GetNSE(); sei++)
-    {
-      qual = TriangleQualityInst (mesh[mesh[sei][0]],
-				  mesh[mesh[sei][1]],
-				  mesh[mesh[sei][2]]);
+    (*testout) << endl << endl;
 
-      cl = int ( (ncl-1e-3) * qual ) + 1;
-      incl.Elem(cl)++;
-    }
+    (*testout) << "Points:           " << mesh.GetNP() << endl;
+    (*testout) << "Surface Elements: " << mesh.GetNSE() << endl;
 
+    (*testout) << endl;
+    (*testout) << "Elements in qualityclasses:" << endl;
+    (*testout).precision(2);
+    for (i = 1; i <= ncl; i++)
+      {
+	(*testout) << setw(4) << double (i-1)/ncl << " - "
+		   << setw(4) << double (i) / ncl << ": "
+		   << incl.Get(i) << endl;
+      }
+  }
 
-  (*testout) << endl << endl;
 
-  (*testout) << "Points:           " << mesh.GetNP() << endl;
-  (*testout) << "Surface Elements: " << mesh.GetNSE() << endl;
+  static double TetElementQuality (const Point3d & p1, const Point3d & p2,
+				   const Point3d & p3, const Point3d & p4)
+  {
+    double vol, l, l4, l5, l6;
 
-  (*testout) << endl;
-  (*testout) << "Elements in qualityclasses:" << endl;
-  (*testout).precision(2);
-  for (i = 1; i <= ncl; i++)
-    {
-      (*testout) << setw(4) << double (i-1)/ncl << " - "
-		 << setw(4) << double (i) / ncl << ": "
-		 << incl.Get(i) << endl;
-    }
-}
 
+    Vec3d v1 = p2 - p1;
+    Vec3d v2 = p3 - p1;
+    Vec3d v3 = p4 - p1;
 
-static double TetElementQuality (const Point3d & p1, const Point3d & p2,
-				 const Point3d & p3, const Point3d & p4)
-{
-  double vol, l, l4, l5, l6;
+    vol = fabs ((Cross (v1, v2) * v3)) / 6;
+    l4 = Dist (p2, p3);
+    l5 = Dist (p2, p4);
+    l6 = Dist (p3, p4);
 
+    l = v1.Length() + v2.Length() + v3.Length() + l4 + l5 + l6;
 
-  Vec3d v1 = p2 - p1;
-  Vec3d v2 = p3 - p1;
-  Vec3d v3 = p4 - p1;
+    if (vol <= 1e-8 * l * l * l) return 1e-10;
 
-  vol = fabs ((Cross (v1, v2) * v3)) / 6;
-  l4 = Dist (p2, p3);
-  l5 = Dist (p2, p4);
-  l6 = Dist (p3, p4);
+    return vol/(l*l*l) * 1832.82;    // 6^4 * sqrt(2)
+  }
 
-  l = v1.Length() + v2.Length() + v3.Length() + l4 + l5 + l6;
 
-  if (vol <= 1e-8 * l * l * l) return 1e-10;
 
-  return vol/(l*l*l) * 1832.82;    // 6^4 * sqrt(2)
-}
 
 
+  double teterrpow = 2;
 
+  double CalcTetBadness (const Point3d & p1, const Point3d & p2,
+			 const Point3d & p3, const Point3d & p4, double h)
+  {
+    double vol, l, ll, lll, ll1, ll2, ll3, ll4, ll5, ll6;
+    double err;
 
+    Vec3d v1 (p1, p2);
+    Vec3d v2 (p1, p3);
+    Vec3d v3 (p1, p4);
 
-double teterrpow = 2;
+    vol = -Determinant (v1, v2, v3) / 6;
 
+    ll1 = v1.Length2();
+    ll2 = v2.Length2();
+    ll3 = v3.Length2();
+    ll4 = Dist2 (p2, p3);
+    ll5 = Dist2 (p2, p4);
+    ll6 = Dist2 (p3, p4);
 
-double CalcTetBadnessNew (const Point3d & p1, const Point3d & p2,
-			  const Point3d & p3, const Point3d & p4, double h)
-{
-  double vol, l, ll, lll, ll1, ll2, ll3, ll4, ll5, ll6;
-  double err;
+    ll = ll1 + ll2 + ll3 + ll4 + ll5 + ll6;
+    l = sqrt (ll);
+    lll = l * ll;
 
-  Vec3d v1 (p1, p2);
-  Vec3d v2 (p1, p3);
-  Vec3d v3 (p1, p4);
+    if (vol <= 1e-24 * lll)
+      return 1e24;
 
-  vol = Determinant (v1, v2, v3) / 6;
+    err = 0.0080187537 * lll / vol;    // sqrt(216) / (6^4 * sqrt(2))
 
-  ll1 = v1.Length2();
-  ll2 = v2.Length2();
-  ll3 = v3.Length2();
-  ll4 = Dist2 (p2, p3);
-  ll5 = Dist2 (p2, p4);
-  ll6 = Dist2 (p3, p4);
+    if (h > 0)
+      err += ll / (h * h) + 
+	h * h * ( 1 / ll1 + 1 / ll2 + 1 / ll3 + 
+		  1 / ll4 + 1 / ll5 + 1 / ll6 ) - 12;
 
-  ll = ll1 + ll2 + ll3 + ll4 + ll5 + ll6;
-  l = sqrt (ll);
-  lll = l * ll;
+    return pow (err, teterrpow);
+  }
 
-  if (vol <= 1e-24 * lll)
-    return 1e24;
 
-  err = (lll) / (1832.82 * vol);    // 6^4 * sqrt(2)
+  double CalcTetBadnessGrad (const Point3d & p1, const Point3d & p2,
+			     const Point3d & p3, const Point3d & p4, double h,
+			     int pi, Vec3d & grad)
+  {
+    double vol, l, ll, lll;
+    double err;
 
-  if (h > 0)
-    err += ll / (h * h) + 
-      h * h * ( 1 / ll1 + 1 / ll2 + 1 / ll3 + 
-		1 / ll4 + 1 / ll5 + 1 / ll6 ) - 12;
+    const Point3d *pp1, *pp2, *pp3, *pp4;
 
-  return pow (err, teterrpow);
-}
+    pp1 = &p1;
+    pp2 = &p2;
+    pp3 = &p3;
+    pp4 = &p4;
+  
+    switch (pi)
+      {
+      case 2:
+	{
+	  swap (pp1, pp2);
+	  swap (pp3, pp4);
+	  break;
+	}
+      case 3:
+	{
+	  swap (pp1, pp3);
+	  swap (pp2, pp4);
+	  break;
+	}
+      case 4:
+	{
+	  swap (pp1, pp4);
+	  swap (pp3, pp2);
+	  break;
+	}
+      }
+  
 
+    Vec3d v1 (*pp1, *pp2);
+    Vec3d v2 (*pp1, *pp3);
+    Vec3d v3 (*pp1, *pp4);
 
+    Vec3d v4 (*pp2, *pp3);
+    Vec3d v5 (*pp2, *pp4);
+    Vec3d v6 (*pp3, *pp4);
 
+    vol = -Determinant (v1, v2, v3) / 6;  
 
+    Vec3d gradvol;
+    Cross (v5, v4, gradvol);
+    gradvol *= (-1.0/6.0);
 
-double CalcTetBadness (const Point3d & p1, const Point3d & p2,
-		       const Point3d & p3, const Point3d & p4, double h)
-{
+
+    double ll1 = v1.Length2();
+    double ll2 = v2.Length2();
+    double ll3 = v3.Length2();
+    double ll4 = v4.Length2();
+    double ll5 = v5.Length2();
+    double ll6 = v6.Length2();
+
+    ll = ll1 + ll2 + ll3 + ll4 + ll5 + ll6;
+    l = sqrt (ll);
+    lll = l * ll;
+
+    if (vol <= 1e-24 * lll)
+      { 
+	grad = Vec3d (0, 0, 0);
+	return 1e24;
+      }
+
+
+
+    Vec3d gradll1 (*pp2, *pp1);
+    Vec3d gradll2 (*pp3, *pp1);
+    Vec3d gradll3 (*pp4, *pp1);
+    gradll1 *= 2;
+    gradll2 *= 2;
+    gradll3 *= 2;
+
+    Vec3d gradll (gradll1);
+    gradll += gradll2;
+    gradll += gradll3;
+
+
+
+    err = 0.0080187537 * lll / vol; 
+
+
+    gradll *= (0.0080187537 * 1.5 * l / vol);
+    Vec3d graderr(gradll);
+    gradvol *= ( -0.0080187537 * lll / (vol * vol) );
+    graderr += gradvol;
+  
+    if (h > 0)
+      {
+	err += ll / (h*h) + 
+	  h*h * ( 1 / ll1 + 1 / ll2 + 1 / ll3 + 
+		  1 / ll4 + 1 / ll5 + 1 / ll6 ) - 12;
+
+	graderr += (1/(h*h) - h*h/(ll1*ll1)) * gradll1;
+	graderr += (1/(h*h) - h*h/(ll2*ll2)) * gradll2;
+	graderr += (1/(h*h) - h*h/(ll3*ll3)) * gradll3;
+	cout << "?";
+      }
+
+    double errpow = pow (err, teterrpow);
+    grad = (teterrpow * errpow / err) * graderr;
+  
+    return errpow;
+  }
+  
+
+
+
+
+  /*
+
+  double CalcTetBadness (const Point3d & p1, const Point3d & p2,
+  const Point3d & p3, const Point3d & p4, double h)
+  {
   double vol, l;
   double err;
 
@@ -257,26 +366,29 @@ double CalcTetBadness (const Point3d & p1, const Point3d & p2,
 
   l = l1 + l2 + l3 + l4 + l5 + l6;
 
+  // just for timing
+  // l += 1e-40 * CalcTetBadnessNew (p1, p2, p3, p4, h);
+
   if (vol <= 1e-24 * l * l * l)
-    { 
-      return 1e24;
-    }
+  { 
+  return 1e24;
+  }
 
   err = (l*l*l) / (1832.82 * vol);    // 6^4 * sqrt(2)
   
   if (h > 0)
-    err += l / h + 
-      h * (1 / l1 + 1/l2 + 1/l3 + 1/l4 + 1/l5 + 1/l6) - 12;
+  err += l / h + 
+  h * (1 / l1 + 1/l2 + 1/l3 + 1/l4 + 1/l5 + 1/l6) - 12;
 
   return pow (err, teterrpow);
-}
+  }
 
 
   
-double CalcTetBadnessGrad (const Point3d & p1, const Point3d & p2,
-			   const Point3d & p3, const Point3d & p4, double h,
-			   int pi, Vec3d & grad)
-{
+  double CalcTetBadnessGrad (const Point3d & p1, const Point3d & p2,
+  const Point3d & p3, const Point3d & p4, double h,
+  int pi, Vec3d & grad)
+  {
   double vol, l;
   double err;
 
@@ -288,26 +400,26 @@ double CalcTetBadnessGrad (const Point3d & p1, const Point3d & p2,
   pp4 = &p4;
   
   switch (pi)
-    {
-    case 2:
-      {
-	Swap (pp1, pp2);
-	Swap (pp3, pp4);
-	break;
-      }
-    case 3:
-      {
-	Swap (pp1, pp3);
-	Swap (pp2, pp4);
-	break;
-      }
-    case 4:
-      {
-	Swap (pp1, pp4);
-	Swap (pp3, pp2);
-	break;
-      }
-    }
+  {
+  case 2:
+  {
+  swap (pp1, pp2);
+  swap (pp3, pp4);
+  break;
+  }
+  case 3:
+  {
+  swap (pp1, pp3);
+  swap (pp2, pp4);
+  break;
+  }
+  case 4:
+  {
+  swap (pp1, pp4);
+  swap (pp3, pp2);
+  break;
+  }
+  }
   
 
   Vec3d v1 (*pp1, *pp2);
@@ -318,11 +430,11 @@ double CalcTetBadnessGrad (const Point3d & p1, const Point3d & p2,
   Vec3d v5 (*pp2, *pp4);
   Vec3d v6 (*pp3, *pp4);
 
-  /*
-  Vec3d n;
-  Cross (v1, v2, n);
-  vol = - (n * v3) / 6;
-  */
+
+  //   Vec3d n;
+  //   Cross (v1, v2, n);
+  //   vol = - (n * v3) / 6;
+
 
   vol = -Determinant (v1, v2, v3) / 6;  
 
@@ -353,10 +465,10 @@ double CalcTetBadnessGrad (const Point3d & p1, const Point3d & p2,
 
 
   if (vol <= 1e-24 * l * l * l)
-    { 
-      grad = Vec3d (0, 0, 0);
-      return 1e24;
-    }
+  { 
+  grad = Vec3d (0, 0, 0);
+  return 1e24;
+  }
 
 
   double c1 = 1.0 / 1832.82;      // 6^4 * sqrt(2)
@@ -369,228 +481,229 @@ double CalcTetBadnessGrad (const Point3d & p1, const Point3d & p2,
   graderr+= gradvol;
   
   if (h > 0)
-    {
-      err += l / h + 
-	h * ( 1 / l1 + 1 / l2 + 1 / l3 + 
-	      1 / l4 + 1 / l5 + 1 / l6 ) - 12;
+  {
+  err += l / h + 
+  h * ( 1 / l1 + 1 / l2 + 1 / l3 + 
+  1 / l4 + 1 / l5 + 1 / l6 ) - 12;
 
-      graderr += (1/h - h/(l1*l1)) * gradl1;
-      graderr += (1/h - h/(l2*l2)) * gradl2;
-      graderr += (1/h - h/(l3*l3)) * gradl3;
-      cout << "?";
-    }
+  graderr += (1/h - h/(l1*l1)) * gradl1;
+  graderr += (1/h - h/(l2*l2)) * gradl2;
+  graderr += (1/h - h/(l3*l3)) * gradl3;
+  cout << "?";
+  }
 
   double errpow = pow (err, teterrpow);
   grad = (teterrpow * errpow / err) * graderr;
   
   return errpow;
-}
+  }
   
+  */
 
 
 
 
   
-/*
-double CalcVolume (const ARRAY<Point3d> & points,
-        const Element & el)
-  {
-  Vec3d v1 = points.Get(el.PNum(2)) - 
-             points.Get(el.PNum(1));
-  Vec3d v2 = points.Get(el.PNum(3)) - 
-             points.Get(el.PNum(1));
-  Vec3d v3 = points.Get(el.PNum(4)) - 
-             points.Get(el.PNum(1)); 
+  /*
+    double CalcVolume (const ARRAY<Point3d> & points,
+    const Element & el)
+    {
+    Vec3d v1 = points.Get(el.PNum(2)) - 
+    points.Get(el.PNum(1));
+    Vec3d v2 = points.Get(el.PNum(3)) - 
+    points.Get(el.PNum(1));
+    Vec3d v3 = points.Get(el.PNum(4)) - 
+    points.Get(el.PNum(1)); 
          
-  return -(Cross (v1, v2) * v3) / 6;	 
-  }  
+    return -(Cross (v1, v2) * v3) / 6;	 
+    }  
   */
 
-double CalcVolume (const ARRAY<Point3d> & points, 
-		   const ARRAY<Element> & elements)
-{
-  double vol;
-  Vec3d v1, v2, v3;
+  double CalcVolume (const ARRAY<Point3d> & points, 
+		     const ARRAY<Element> & elements)
+  {
+    double vol;
+    Vec3d v1, v2, v3;
   
-  vol = 0;
-  for (int i = 0; i < elements.Size(); i++)
-    {
-      v1 = points.Get(elements[i][1]) - points.Get(elements[i][0]);
-      v2 = points.Get(elements[i][2]) - points.Get(elements[i][0]);
-      v3 = points.Get(elements[i][3]) - points.Get(elements[i][0]);
-      vol -= (Cross (v1, v2) * v3) / 6;	 
-    }
-  return vol;
-}
+    vol = 0;
+    for (int i = 0; i < elements.Size(); i++)
+      {
+	v1 = points.Get(elements[i][1]) - points.Get(elements[i][0]);
+	v2 = points.Get(elements[i][2]) - points.Get(elements[i][0]);
+	v3 = points.Get(elements[i][3]) - points.Get(elements[i][0]);
+	vol -= (Cross (v1, v2) * v3) / 6;	 
+      }
+    return vol;
+  }
 
   
   
 
-void MeshQuality3d (const Mesh & mesh, ARRAY<int> * inclass)
-{ 
-  int ncl = 20;
-  signed int cl;
-  ARRAY<INDEX> incl(ncl);
-  INDEX i;
-  double qual;
-  double sum = 0;
-  int nontet  = 0;
+  void MeshQuality3d (const Mesh & mesh, ARRAY<int> * inclass)
+  { 
+    int ncl = 20;
+    signed int cl;
+    ARRAY<INDEX> incl(ncl);
+    INDEX i;
+    double qual;
+    double sum = 0;
+    int nontet  = 0;
 
-  for (i = 1; i <= incl.Size(); i++)
-    incl.Elem(i) = 0;
+    for (i = 1; i <= incl.Size(); i++)
+      incl.Elem(i) = 0;
 
-  for (ElementIndex ei = 0; ei < mesh.GetNE(); ei++)
-    {
-      if (mesh[ei].GetType() != TET)
-	{
-	  nontet++;
-	  continue;
-	}
+    for (ElementIndex ei = 0; ei < mesh.GetNE(); ei++)
+      {
+	if (mesh[ei].GetType() != TET)
+	  {
+	    nontet++;
+	    continue;
+	  }
+
+	qual = TetElementQuality (mesh.Point(mesh[ei][0]),
+				  mesh.Point(mesh[ei][1]),
+				  mesh.Point(mesh[ei][2]),
+				  mesh.Point(mesh[ei][3]));
+
+	if (qual > 1) qual = 1;
+	cl = int (ncl * qual ) + 1;
+     
+	if (cl < 1) cl = 1; 
+	if (cl > ncl) cl = ncl;
 
-      qual = TetElementQuality (mesh.Point(mesh[ei][0]),
-				mesh.Point(mesh[ei][1]),
-				mesh.Point(mesh[ei][2]),
-				mesh.Point(mesh[ei][3]));
+	incl.Elem(cl)++;
+	if (inclass) (*inclass)[ei] = cl;
+	sum += 1/qual;
+      }
 
-      if (qual > 1) qual = 1;
-      cl = int (ncl * qual ) + 1;
-     
-      if (cl < 1) cl = 1; 
-      if (cl > ncl) cl = ncl;
-
-      incl.Elem(cl)++;
-      if (inclass) (*inclass)[ei] = cl;
-      sum += 1/qual;
-    }
-
-  (*testout) << endl << endl;
-  (*testout) << "Points:           " << mesh.GetNP() << endl;
-  (*testout) << "Volume Elements:  " << mesh.GetNE() << endl;
-  if (nontet)
-    (*testout) << nontet << " non tetrahedral elements" << endl;
-  (*testout) << endl;
-
-  (*testout) << "Volume elements in qualityclasses:" << endl;
-  (*testout).precision(2);
-  for (i = 1; i <= ncl; i++)
-    {
-      (*testout) << setw(4) << double (i-1)/ncl << " - "
-		 << setw(4) << double (i) / ncl << ": "
-		 << incl.Get(i) << endl;
-    }
-  (*testout) << "total error: " << sum << endl;
-}
+    (*testout) << endl << endl;
+    (*testout) << "Points:           " << mesh.GetNP() << endl;
+    (*testout) << "Volume Elements:  " << mesh.GetNE() << endl;
+    if (nontet)
+      (*testout) << nontet << " non tetrahedral elements" << endl;
+    (*testout) << endl;
+
+    (*testout) << "Volume elements in qualityclasses:" << endl;
+    (*testout).precision(2);
+    for (i = 1; i <= ncl; i++)
+      {
+	(*testout) << setw(4) << double (i-1)/ncl << " - "
+		   << setw(4) << double (i) / ncl << ": "
+		   << incl.Get(i) << endl;
+      }
+    (*testout) << "total error: " << sum << endl;
+  }
 
 
-void SaveEdges (const Mesh & mesh, const char * geomfile, double h, char * filename)
-{
-  ofstream of (filename);
-  int i;
-  const Segment * seg;
+  void SaveEdges (const Mesh & mesh, const char * geomfile, double h, char * filename)
+  {
+    ofstream of (filename);
+    int i;
+    const Segment * seg;
   
-  of << "edges" << endl;
-  of << geomfile << endl;
-  of << h << endl;
-
-  of << mesh.GetNP() << endl;
-  for (i = 1; i <= mesh.GetNP(); i++)
-    of << mesh.Point(i).X() << " "
-       << mesh.Point(i).Y() << " "
-       << mesh.Point(i).Z() << "\n";
+    of << "edges" << endl;
+    of << geomfile << endl;
+    of << h << endl;
+
+    of << mesh.GetNP() << endl;
+    for (i = 1; i <= mesh.GetNP(); i++)
+      of << mesh.Point(i).X() << " "
+	 << mesh.Point(i).Y() << " "
+	 << mesh.Point(i).Z() << "\n";
     
-  of << 2 * mesh.GetNSeg() << endl;
-  for (i = 1; i <= mesh.GetNSeg(); i++)
-    {
-      seg = &mesh.LineSegment(i);
+    of << 2 * mesh.GetNSeg() << endl;
+    for (i = 1; i <= mesh.GetNSeg(); i++)
+      {
+	seg = &mesh.LineSegment(i);
 
-      of << seg->p2 << " " << seg->p1 << " " << seg->si << "\n";
-    }
+	of << seg->p2 << " " << seg->p1 << " " << seg->si << "\n";
+      }
    
-}
+  }
 
 
-void SaveSurfaceMesh (const Mesh & mesh,
-		      double h,
-		      char * filename)
+  void SaveSurfaceMesh (const Mesh & mesh,
+			double h,
+			char * filename)
 
-{
-  INDEX i;
+  {
+    INDEX i;
 
-  ofstream outfile(filename);
+    ofstream outfile(filename);
 
-  outfile << "surfacemesh" << endl;
-  outfile << h << endl;
+    outfile << "surfacemesh" << endl;
+    outfile << h << endl;
 
-  outfile << mesh.GetNP() << endl;
-  for (i = 1; i <= mesh.GetNP(); i++)
-    outfile << mesh.Point(i).X() << " "
-            << mesh.Point(i).Y() << " "
-            << mesh.Point(i).Z() << endl;
+    outfile << mesh.GetNP() << endl;
+    for (i = 1; i <= mesh.GetNP(); i++)
+      outfile << mesh.Point(i).X() << " "
+	      << mesh.Point(i).Y() << " "
+	      << mesh.Point(i).Z() << endl;
 
   
 
-  outfile << mesh.GetNSE() << endl;
-  for (i = 1; i <= mesh.GetNSE(); i++)
-    {
-      const Element2d & el = mesh.SurfaceElement(i);
-
-      if (mesh.GetFaceDescriptor(el.GetIndex()).DomainOut() == 0)
-	outfile << mesh.SurfaceElement(i).PNum(1) << " "
-		<< mesh.SurfaceElement(i).PNum(2) << " "
-		<< mesh.SurfaceElement(i).PNum(3) << endl;
-      if (mesh.GetFaceDescriptor(el.GetIndex()).DomainIn() == 0)
-	outfile << mesh.SurfaceElement(i).PNum(1) << " "
-		<< mesh.SurfaceElement(i).PNum(3) << " "
-		<< mesh.SurfaceElement(i).PNum(2) << endl;
-    }
-}
+    outfile << mesh.GetNSE() << endl;
+    for (i = 1; i <= mesh.GetNSE(); i++)
+      {
+	const Element2d & el = mesh.SurfaceElement(i);
+
+	if (mesh.GetFaceDescriptor(el.GetIndex()).DomainOut() == 0)
+	  outfile << mesh.SurfaceElement(i).PNum(1) << " "
+		  << mesh.SurfaceElement(i).PNum(2) << " "
+		  << mesh.SurfaceElement(i).PNum(3) << endl;
+	if (mesh.GetFaceDescriptor(el.GetIndex()).DomainIn() == 0)
+	  outfile << mesh.SurfaceElement(i).PNum(1) << " "
+		  << mesh.SurfaceElement(i).PNum(3) << " "
+		  << mesh.SurfaceElement(i).PNum(2) << endl;
+      }
+  }
 
 
 #ifdef OLD
-void Save2DMesh (
-		 const Mesh & mesh2d,
-		 const ARRAY<SplineSegment *> * splines,
-		 ostream & outfile)
+  void Save2DMesh (
+		   const Mesh & mesh2d,
+		   const ARRAY<SplineSegment *> * splines,
+		   ostream & outfile)
 
-{
-  int i, j;
-  outfile.precision (6);
+  {
+    int i, j;
+    outfile.precision (6);
   
-  outfile << "areamesh2" << endl;
+    outfile << "areamesh2" << endl;
 
 
-  outfile << endl;
-  outfile << mesh2d.GetNSeg() << endl;
-  for (i = 1; i <= mesh2d.GetNSeg(); i++)
-    outfile << mesh2d.LineSegment(i).si << "        "
-	    << mesh2d.LineSegment(i).p1 << " "
-	    << mesh2d.LineSegment(i).p2 << "  " << endl;
+    outfile << endl;
+    outfile << mesh2d.GetNSeg() << endl;
+    for (i = 1; i <= mesh2d.GetNSeg(); i++)
+      outfile << mesh2d.LineSegment(i).si << "        "
+	      << mesh2d.LineSegment(i).p1 << " "
+	      << mesh2d.LineSegment(i).p2 << "  " << endl;
   
 
-  outfile << mesh2d.GetNSE() << endl;
-  for (i = 1; i <= mesh2d.GetNSE(); i++)
-    {
-      outfile << mesh2d.SurfaceElement(i).GetIndex() << "         ";
-      outfile << mesh2d.SurfaceElement(i).GetNP() << " ";
-      for (j = 1; j <= mesh2d.SurfaceElement(i).GetNP(); j++)
-	outfile << mesh2d.SurfaceElement(i).PNum(j) << " ";
-      outfile << endl;
-    }
-
-  outfile << mesh2d.GetNP() << endl;
-  for (i = 1; i <= mesh2d.GetNP(); i++)
-    outfile << mesh2d.Point(i).X() << " "
-            << mesh2d.Point(i).Y() << endl;
-
-  if (splines)
-    {
-      outfile << splines->Size() << endl;
-      for (i = 1; i <= splines->Size(); i++)
-	splines->Get(i) -> PrintCoeff (outfile);
-    }
-  else
-    outfile << "0" << endl;
-}
+    outfile << mesh2d.GetNSE() << endl;
+    for (i = 1; i <= mesh2d.GetNSE(); i++)
+      {
+	outfile << mesh2d.SurfaceElement(i).GetIndex() << "         ";
+	outfile << mesh2d.SurfaceElement(i).GetNP() << " ";
+	for (j = 1; j <= mesh2d.SurfaceElement(i).GetNP(); j++)
+	  outfile << mesh2d.SurfaceElement(i).PNum(j) << " ";
+	outfile << endl;
+      }
+
+    outfile << mesh2d.GetNP() << endl;
+    for (i = 1; i <= mesh2d.GetNP(); i++)
+      outfile << mesh2d.Point(i).X() << " "
+	      << mesh2d.Point(i).Y() << endl;
+
+    if (splines)
+      {
+	outfile << splines->Size() << endl;
+	for (i = 1; i <= splines->Size(); i++)
+	  splines->Get(i) -> PrintCoeff (outfile);
+      }
+    else
+      outfile << "0" << endl;
+  }
 #endif
 
 
@@ -600,266 +713,266 @@ void Save2DMesh (
 
 
 
-void SaveVolumeMesh (const Mesh & mesh, 
-		     const CSGeometry & geometry,
-		     char * filename)
-{
-  INDEX i;
+  void SaveVolumeMesh (const Mesh & mesh, 
+		       const CSGeometry & geometry,
+		       char * filename)
+  {
+    INDEX i;
 
-  ofstream outfile(filename);
-  outfile << "volumemesh" << endl;
+    ofstream outfile(filename);
+    outfile << "volumemesh" << endl;
 
-  outfile << mesh.GetNSE() << endl;
-  for (i = 1; i <= mesh.GetNSE(); i++)
-    {
-      if (mesh.SurfaceElement(i).GetIndex())
-	outfile << mesh.GetFaceDescriptor(mesh.SurfaceElement(i).GetIndex ()).SurfNr()
-		<< "\t";
-      else
-	outfile << "0" << "\t";
-      outfile << mesh.SurfaceElement(i)[0] << " "
-	      << mesh.SurfaceElement(i)[1] << " "
-	      << mesh.SurfaceElement(i)[2] << endl;
-    }
-  outfile << mesh.GetNE() << endl;
-  for (ElementIndex ei = 0; ei < mesh.GetNE(); ei++)
-    outfile << mesh[ei].GetIndex() << "\t"
-	    << mesh[ei][0] << " " << mesh[ei][1] << " "
-            << mesh[ei][2] << " " << mesh[ei][3] << endl;
-
-  outfile << mesh.GetNP() << endl;
-  for (i = 1; i <= mesh.GetNP(); i++)
-    outfile << mesh.Point(i).X() << " "
-            << mesh.Point(i).Y() << " "
-            << mesh.Point(i).Z() << endl;
+    outfile << mesh.GetNSE() << endl;
+    for (i = 1; i <= mesh.GetNSE(); i++)
+      {
+	if (mesh.SurfaceElement(i).GetIndex())
+	  outfile << mesh.GetFaceDescriptor(mesh.SurfaceElement(i).GetIndex ()).SurfNr()
+		  << "\t";
+	else
+	  outfile << "0" << "\t";
+	outfile << mesh.SurfaceElement(i)[0] << " "
+		<< mesh.SurfaceElement(i)[1] << " "
+		<< mesh.SurfaceElement(i)[2] << endl;
+      }
+    outfile << mesh.GetNE() << endl;
+    for (ElementIndex ei = 0; ei < mesh.GetNE(); ei++)
+      outfile << mesh[ei].GetIndex() << "\t"
+	      << mesh[ei][0] << " " << mesh[ei][1] << " "
+	      << mesh[ei][2] << " " << mesh[ei][3] << endl;
+
+    outfile << mesh.GetNP() << endl;
+    for (i = 1; i <= mesh.GetNP(); i++)
+      outfile << mesh.Point(i).X() << " "
+	      << mesh.Point(i).Y() << " "
+	      << mesh.Point(i).Z() << endl;
 
 #ifdef SOLIDGEOM
-  outfile << geometry.GetNSurf() << endl;
-  for (i = 1; i <= geometry.GetNSurf(); i++)
-    geometry.GetSurface(i) -> Print (outfile);
+    outfile << geometry.GetNSurf() << endl;
+    for (i = 1; i <= geometry.GetNSurf(); i++)
+      geometry.GetSurface(i) -> Print (outfile);
 #endif
-}
+  }
 
 
 
 
-int CheckCode ()
-{
-  return 1;
+  int CheckCode ()
+  {
+    return 1;
 
-  /*
-    char st[100];
-    ifstream ist("pw");
+    /*
+      char st[100];
+      ifstream ist("pw");
 
-    if (!ist.good()) return 0;
-    ist >> st;
-    if (strcmp (st, "JKULinz") == 0) return 1;
-    return 0;
+      if (!ist.good()) return 0;
+      ist >> st;
+      if (strcmp (st, "JKULinz") == 0) return 1;
+      return 0;
     */
-}
+  }
 
 
 
-/* ******************** CheckMesh ******************************* */
+  /* ******************** CheckMesh ******************************* */
 
-/// Checks, whether mesh contains a valid 3d mesh
-int CheckMesh3D (const Mesh & mesh)
-{
-  INDEX_3_HASHTABLE<int> faceused(mesh.GetNE()/3);
-  INDEX i;
-  int j, k, l;
-  INDEX_3 i3;
-  int ok = 1;
-  ElementIndex ei;
-
-  for (i = 1; i <= mesh.GetNSE(); i++)
-    {
-      const Element2d & el = mesh.SurfaceElement(i);
+  /// Checks, whether mesh contains a valid 3d mesh
+  int CheckMesh3D (const Mesh & mesh)
+  {
+    INDEX_3_HASHTABLE<int> faceused(mesh.GetNE()/3);
+    INDEX i;
+    int j, k, l;
+    INDEX_3 i3;
+    int ok = 1;
+    ElementIndex ei;
+
+    for (i = 1; i <= mesh.GetNSE(); i++)
+      {
+	const Element2d & el = mesh.SurfaceElement(i);
       
-      if (mesh.GetFaceDescriptor(el.GetIndex()).DomainIn() == 0 ||
-	  mesh.GetFaceDescriptor(el.GetIndex()).DomainOut() == 0)
-	{
-	  for (j = 1; j <= 3; j++)
-	    i3.I(j) = el.PNum(j);
+	if (mesh.GetFaceDescriptor(el.GetIndex()).DomainIn() == 0 ||
+	    mesh.GetFaceDescriptor(el.GetIndex()).DomainOut() == 0)
+	  {
+	    for (j = 1; j <= 3; j++)
+	      i3.I(j) = el.PNum(j);
 	  
-	  i3.Sort();
-	  faceused.Set (i3, 1);
-	}
-    }
-  
-  for (ei = 0; ei < mesh.GetNE(); ei++)
-    {
-      const Element & el = mesh[ei];
-
-      for (j = 1; j <= 4; j++)
-	{
-	  l = 0;
-	  for (k = 1; k <= 4; k++)
-	    {
-	      if (j != k)
-		{
-		  l++;
-		  i3.I(l) = el.PNum(k);
-		}
-	    }
-
-	  i3.Sort();
-	  if (faceused.Used(i3))
-	    faceused.Set(i3, faceused.Get(i3)+1);
-	  else
+	    i3.Sort();
 	    faceused.Set (i3, 1);
-	}
-    }
+	  }
+      }
+  
+    for (ei = 0; ei < mesh.GetNE(); ei++)
+      {
+	const Element & el = mesh[ei];
+
+	for (j = 1; j <= 4; j++)
+	  {
+	    l = 0;
+	    for (k = 1; k <= 4; k++)
+	      {
+		if (j != k)
+		  {
+		    l++;
+		    i3.I(l) = el.PNum(k);
+		  }
+	      }
+
+	    i3.Sort();
+	    if (faceused.Used(i3))
+	      faceused.Set(i3, faceused.Get(i3)+1);
+	    else
+	      faceused.Set (i3, 1);
+	  }
+      }
 
 
-  for (i = 1; i <= mesh.GetNSE(); i++)
-    {
-      const Element2d & el = mesh.SurfaceElement(i);
+    for (i = 1; i <= mesh.GetNSE(); i++)
+      {
+	const Element2d & el = mesh.SurfaceElement(i);
 
-      for (j = 1; j <= 3; j++)
-	i3.I(j) = el.PNum(j);
+	for (j = 1; j <= 3; j++)
+	  i3.I(j) = el.PNum(j);
       
-      i3.Sort();
-      k = faceused.Get (i3);
-      if (k != 2)
-	{
-	  ok = 0;
-	  (*testout) << "face " << i << " with points " 
-		     << i3.I1() << "-" << i3.I2() << "-" << i3.I3() 
-		     << " has " << k << " elements" << endl;
-	}
-    }
+	i3.Sort();
+	k = faceused.Get (i3);
+	if (k != 2)
+	  {
+	    ok = 0;
+	    (*testout) << "face " << i << " with points " 
+		       << i3.I1() << "-" << i3.I2() << "-" << i3.I3() 
+		       << " has " << k << " elements" << endl;
+	  }
+      }
   
-  for (ei = 0; ei < mesh.GetNE(); ei++)
-    {
-      const Element & el = mesh[ei];
-
-      for (j = 1; j <= 4; j++)
-	{
-	  l = 0;
-	  for (k = 1; k <= 4; k++)
-	    {
-	      if (j != k)
-		{
-		  l++;
-		  i3.I(l) = el.PNum(k);
-		}
-	    }
-
-	  i3.Sort();
-	  k = faceused.Get(i3);
-	  if (k != 2)
-	    {
-	      ok = 0;
-	      (*testout) << "element " << ei << " with face " 
-			 << i3.I1() << "-" << i3.I2() << "-"
-			 << i3.I3() 
-			 << " has " << k << " elements" << endl;
-	    }
-	}
-    }
+    for (ei = 0; ei < mesh.GetNE(); ei++)
+      {
+	const Element & el = mesh[ei];
+
+	for (j = 1; j <= 4; j++)
+	  {
+	    l = 0;
+	    for (k = 1; k <= 4; k++)
+	      {
+		if (j != k)
+		  {
+		    l++;
+		    i3.I(l) = el.PNum(k);
+		  }
+	      }
+
+	    i3.Sort();
+	    k = faceused.Get(i3);
+	    if (k != 2)
+	      {
+		ok = 0;
+		(*testout) << "element " << ei << " with face " 
+			   << i3.I1() << "-" << i3.I2() << "-"
+			   << i3.I3() 
+			   << " has " << k << " elements" << endl;
+	      }
+	  }
+      }
 
 
 
 
 
-  /*
-    for (i = 1; i <= faceused.GetNBags(); i++)
-    for (j = 1; j <= faceused.GetBagSize(i); j++)
-    {
-    faceused.GetData(i, j, i3, k);
-    if (k != 2)
-    {
-    (*testout) << "Face: " << i3.I1() << "-" 
-    << i3.I2() << "-" << i3.I3() << " has " 
-    << k << " Faces " << endl;
-    cerr << "Face Error" << endl;
-    ok = 0;
-    }
-    }
+    /*
+      for (i = 1; i <= faceused.GetNBags(); i++)
+      for (j = 1; j <= faceused.GetBagSize(i); j++)
+      {
+      faceused.GetData(i, j, i3, k);
+      if (k != 2)
+      {
+      (*testout) << "Face: " << i3.I1() << "-" 
+      << i3.I2() << "-" << i3.I3() << " has " 
+      << k << " Faces " << endl;
+      cerr << "Face Error" << endl;
+      ok = 0;
+      }
+      }
     */
 
 
-  if (!ok)
-    {
-      (*testout) << "surfelements: " << endl;
-      for (i = 1; i <= mesh.GetNSE(); i++)
-	{
-	  const Element2d & el = mesh.SurfaceElement(i);
-	  (*testout) << setw(5) << i << ":" 
-		     << setw(6) << el.GetIndex() 
-		     << setw(6) << el.PNum(1) 
-		     << setw(4) << el.PNum(2) 
-		     << setw(4) << el.PNum(3)  << endl;
-	}
-      (*testout) << "volelements: " << endl;
-      for (ElementIndex ei = 0; ei < mesh.GetNE(); ei++)
-	{
-	  const Element & el = mesh[ei];
-	  (*testout) << setw(5) << i << ":" 
-		     << setw(6) << el.GetIndex() 
-		     << setw(6) << el[0] << setw(4) << el[1]
-		     << setw(4) << el[2] << setw(4) << el[3] << endl;
-	}
-    }
+    if (!ok)
+      {
+	(*testout) << "surfelements: " << endl;
+	for (i = 1; i <= mesh.GetNSE(); i++)
+	  {
+	    const Element2d & el = mesh.SurfaceElement(i);
+	    (*testout) << setw(5) << i << ":" 
+		       << setw(6) << el.GetIndex() 
+		       << setw(6) << el.PNum(1) 
+		       << setw(4) << el.PNum(2) 
+		       << setw(4) << el.PNum(3)  << endl;
+	  }
+	(*testout) << "volelements: " << endl;
+	for (ElementIndex ei = 0; ei < mesh.GetNE(); ei++)
+	  {
+	    const Element & el = mesh[ei];
+	    (*testout) << setw(5) << i << ":" 
+		       << setw(6) << el.GetIndex() 
+		       << setw(6) << el[0] << setw(4) << el[1]
+		       << setw(4) << el[2] << setw(4) << el[3] << endl;
+	  }
+      }
 
 
-  return ok;
-}
+    return ok;
+  }
 
 
 
-void RemoveProblem (Mesh & mesh)
-{
-  int i, j, k;
+  void RemoveProblem (Mesh & mesh)
+  {
+    int i, j, k;
   
-  mesh.FindOpenElements();
-  int np = mesh.GetNP();
+    mesh.FindOpenElements();
+    int np = mesh.GetNP();
 
-  BitArrayChar<PointIndex::BASE> ppoints(np);
+    BitArrayChar<PointIndex::BASE> ppoints(np);
   
-  int ndom = mesh.GetNDomains();
+    int ndom = mesh.GetNDomains();
 
-  PrintMessage (3, "Elements before Remove: ", mesh.GetNE());
-  for (k = 1; k <= ndom; k++)
-    {
-      ppoints.Clear();
+    PrintMessage (3, "Elements before Remove: ", mesh.GetNE());
+    for (k = 1; k <= ndom; k++)
+      {
+	ppoints.Clear();
       
-      for (i = 1; i <= mesh.GetNOpenElements(); i++)
-	{
-	  const Element2d & sel = mesh.OpenElement(i);
-	  if (sel.GetIndex() == k)
-	    {
-	      for (j = 1; j <= sel.GetNP(); j++)
-		ppoints.Set (sel.PNum(j));
-	    }
-	}
-
-      for (ElementIndex ei = 0; ei < mesh.GetNE(); ei++)
-	{
-	  const Element & el = mesh[ei];
-	  if (el.GetIndex() == k)
-	    {
-	      int todel = 0;
-	      for (j = 0; j < el.GetNP(); j++)
-		if (ppoints.Test (el[j]))
-		  todel = 1;
+	for (i = 1; i <= mesh.GetNOpenElements(); i++)
+	  {
+	    const Element2d & sel = mesh.OpenElement(i);
+	    if (sel.GetIndex() == k)
+	      {
+		for (j = 1; j <= sel.GetNP(); j++)
+		  ppoints.Set (sel.PNum(j));
+	      }
+	  }
+
+	for (ElementIndex ei = 0; ei < mesh.GetNE(); ei++)
+	  {
+	    const Element & el = mesh[ei];
+	    if (el.GetIndex() == k)
+	      {
+		int todel = 0;
+		for (j = 0; j < el.GetNP(); j++)
+		  if (ppoints.Test (el[j]))
+		    todel = 1;
 	      
-	      if (el.GetNP() != 4)
-		todel = 0;
+		if (el.GetNP() != 4)
+		  todel = 0;
 	      
-	      if (todel)
-		{
-		  mesh[ei].Delete();
-		  // ei--;
-		}
-	    }
-	}
-    }
+		if (todel)
+		  {
+		    mesh[ei].Delete();
+		    // ei--;
+		  }
+	      }
+	  }
+      }
   
-  mesh.Compress();
-  PrintMessage (3, "Elements after Remove: ", mesh.GetNE());
-}
+    mesh.Compress();
+    PrintMessage (3, "Elements after Remove: ", mesh.GetNE());
+  }
 
 
 }
diff --git a/Netgen/libsrc/meshing/meshtool.hpp b/Netgen/libsrc/meshing/meshtool.hpp
index 397528876f..1baafe40ff 100644
--- a/Netgen/libsrc/meshing/meshtool.hpp
+++ b/Netgen/libsrc/meshing/meshtool.hpp
@@ -44,6 +44,7 @@ void SaveVolumeMesh (const Mesh & mesh,
 ///
 extern int CheckCode ();
 
+
 ///
 extern double CalcTetBadness (const Point3d & p1, 
 			      const Point3d & p2,
diff --git a/Netgen/libsrc/meshing/meshtype.cpp b/Netgen/libsrc/meshing/meshtype.cpp
index 7a8bcb513c..3ed65c5857 100644
--- a/Netgen/libsrc/meshing/meshtype.cpp
+++ b/Netgen/libsrc/meshing/meshtype.cpp
@@ -6,6 +6,8 @@
 namespace netgen
 {
 
+
+
   MultiPointGeomInfo :: MultiPointGeomInfo()
   {
     cnt = 0;
@@ -14,9 +16,7 @@ namespace netgen
   int MultiPointGeomInfo :: 
   AddPointGeomInfo (const PointGeomInfo & gi)
   {
-    int k;
-
-    for (k = 0; k < cnt; k++)
+    for (int k = 0; k < cnt; k++)
       if (mgi[k].trignum == gi.trignum)
 	return 0;
   
@@ -26,12 +26,10 @@ namespace netgen
 	cnt++;
 	return 0;
       }
-    // #ifdef DEVELOP
-    cout << "Please Increase MPGI - Size" << endl;
-    // #endif
-    return 1;
-  }
 
+    throw NgException ("Please report error: MPGI Size too small\n");
+  }
+  
 
   void MultiPointGeomInfo :: 
   Init ()
@@ -50,33 +48,73 @@ namespace netgen
 
   Segment :: Segment() 
   {
-    p1 = p2 = 0; 
-    geominfo[0].trignum=0; 
-    geominfo[1].trignum=0; 
-    surfnr1 = surfnr2 = 0;
-    edgenr = 0;
+    p1 = -1;
+    p2 = -1; 
+    edgenr = -1;
 
-    singedge = 0;
+    singedge_left = 0;
+    singedge_right = 0;
+    seginfo = 0;
+
+    si = -1;
+
+    domin = -1;
+    domout = -1;
+    tlosurf = -1;
+
+    surfnr1 = -1;
+    surfnr2 = -1;
+    pmid = -1;
+    meshdocval = 0;
+    /*
+    geominfo[0].trignum=-1; 
+    geominfo[1].trignum=-1; 
 
     epgeominfo[0].edgenr = 1;
     epgeominfo[0].dist = 0;
     epgeominfo[1].edgenr = 1;
     epgeominfo[1].dist = 0;
+    */
+  }    
+
+
 
-    pmid = 0;
+  Segment& Segment::operator=(const Segment & other)
+  {
+    p1 = other.p1;
+    p2 = other.p2;
+    edgenr = other.edgenr;
+    singedge_left = other.singedge_left;
+    singedge_right = other.singedge_right;
+    seginfo = other.seginfo;
+    si = other.si;
+    domin = other.domin;
+    domout = other.domout;
+    tlosurf = other.tlosurf;
+    geominfo[0] = other.geominfo[0];
+    geominfo[1] = other.geominfo[1];
+    surfnr1 = other.surfnr1;
+    surfnr2 = other.surfnr2;
+    epgeominfo[0] = other.epgeominfo[0];
+    epgeominfo[1] = other.epgeominfo[1];
+    pmid = other.pmid;
+    meshdocval = other.meshdocval;
+
+    return *this;
   }
 
+
   ostream & operator<<(ostream  & s, const Segment & seg)
   {
     s << seg.p1 << "(gi=" << seg.geominfo[0].trignum << ") - "
-      << seg.p2 << "(gi=" << seg.geominfo[1].trignum << ")";
+      << seg.p2 << "(gi=" << seg.geominfo[1].trignum << ")"
+      << " si = " << seg.si;
     return s;
   }
 
   Element2d :: Element2d (int anp)
   { 
-    int i;
-    for (i = 0; i < 6; i++)
+    for (int i = 0; i < ELEMENT2D_MAXPOINTS; i++)
       {
 	pnum[i] = 0;
 	geominfo[i].trignum = 0;
@@ -90,15 +128,15 @@ namespace netgen
       case 3: typ = TRIG; break;
       case 4: typ = QUAD; break;
       case 6: typ = TRIG6; break;
+      case 8: typ = QUAD8; break;
       }
     order = 1;
+    refflag = 1;
   } 
 
-
   Element2d :: Element2d (ELEMENT_TYPE atyp)
   { 
-    int i;
-    for (i = 0; i < 6; i++)
+    for (int i = 0; i < ELEMENT2D_MAXPOINTS; i++)
       {
 	pnum[i] = 0;
 	geominfo[i].trignum = 0;
@@ -110,6 +148,7 @@ namespace netgen
     badel = 0;
     deleted = 0;
     order = 1;
+    refflag = 1;
   } 
 
 
@@ -126,7 +165,7 @@ namespace netgen
   pnum[4] = 0;
   pnum[5] = 0;
   
-  for (int i = 0; i < 6; i++)
+  for (int i = 0; i < ELEMENT2D_MAXPOINTS; i++)
     geominfo[i].trignum = 0;
   index = 0;
   badel = 0;
@@ -147,7 +186,7 @@ Element2d :: Element2d (int pi1, int pi2, int pi3, int pi4)
   pnum[4] = 0;
   pnum[5] = 0;
   
-  for (int i = 0; i < 6; i++)
+  for (int i = 0; i < ELEMENT2D_MAXPOINTS; i++)
     geominfo[i].trignum = 0;
   index = 0;
   badel = 0;
@@ -157,6 +196,7 @@ Element2d :: Element2d (int pi1, int pi2, int pi3, int pi4)
 }
 
 
+/*
 void Element2d :: SetType (ELEMENT_TYPE atyp)
 {
   typ = atyp;
@@ -170,7 +210,7 @@ void Element2d :: SetType (ELEMENT_TYPE atyp)
       PrintSysError ("Element2d::SetType, illegal type ", typ);
     }
 }
-
+*/
 
 
 void Element2d :: GetBox (const T_POINTS & points, Box3d & box) const
@@ -244,13 +284,12 @@ void Element2d :: NormalizeNumbering2 ()
     }
   else
     {
-      int i;
       int mini = 1;
-      for (i = 2; i <= GetNP(); i++)
+      for (int i = 2; i <= GetNP(); i++)
 	if (PNum(i) < PNum(mini)) mini = i;
       
       Element2d hel = (*this);
-      for (i = 1; i <= GetNP(); i++)
+      for (int i = 1; i <= GetNP(); i++)
 	PNum(i) = hel.PNumMod (i+mini-1);
     }
 }
@@ -336,14 +375,16 @@ GetTransformation (int ip, class DenseMatrix & pmat,
 {
   int np = GetNP();
 
+#ifdef DEBUG
   if (pmat.Width() != np || pmat.Height() != 2)
     {
       (*testout) << "GetTransofrmation: pmat doesn't fit" << endl;
       return;
     }
+#endif
 
   ComputeIntegrationPointData ();
-  DenseMatrix * dshapep = 0;
+  DenseMatrix * dshapep;
   switch (typ)
     {
     case TRIG: dshapep = &ipdtrig.Get(ip)->dshape; break;
@@ -425,15 +466,17 @@ void Element2d ::
 GetPointMatrix (const ARRAY<Point2d> & points,
 		DenseMatrix & pmat) const
 {
-  int i;
   int np = GetNP();
+
+#ifdef DEBUG
   if (pmat.Width() != np || pmat.Height() != 2)
     {
       cerr << "Element::GetPointMatrix: sizes don't fit" << endl;
       return;
     }
+#endif
   
-  for (i = 1; i <= np; i++)
+  for (int i = 1; i <= np; i++)
     {
       const Point2d & p = points.Get(PNum(i));
       pmat.Elem(1, i) = p.X();
@@ -480,13 +523,94 @@ double Element2d :: CalcJacobianBadness (const ARRAY<Point2d> & points) const
 }
 
 
+
+static const int qip_table[4][4] =
+  { { 0, 1, 0, 3 },
+    { 0, 1, 1, 2 },
+    { 3, 2, 0, 3 },
+    { 3, 2, 1, 2 }
+  };
+
 double Element2d :: 
 CalcJacobianBadnessDirDeriv (const ARRAY<Point2d> & points,
 			     int pi, Vec2d & dir, double & dd) const
 {
-  int i, j, k, l;
+  if (typ == QUAD)
+    {
+      Mat<2,2> trans, dtrans;
+      Mat<2,4> vmat, pmat;
+      
+      for (int j = 0; j < 4; j++)
+	{
+	  const Point2d & p = points.Get( (*this)[j] );
+	  pmat(0, j) = p.X();
+	  pmat(1, j) = p.Y();
+	}
+
+      vmat = 0.0;
+      vmat(0, pi-1) = dir.X();
+      vmat(1, pi-1) = dir.Y();
+      
+      double err = 0;
+      dd = 0;
+
+      for (int i = 0; i < 4; i++)
+	{
+	  int ix1 = qip_table[i][0];
+	  int ix2 = qip_table[i][1];
+	  int iy1 = qip_table[i][2];
+	  int iy2 = qip_table[i][3];
+	      
+	  trans(0,0) = pmat(0, ix2) - pmat(0,ix1);
+	  trans(1,0) = pmat(1, ix2) - pmat(1,ix1);
+	  trans(0,1) = pmat(0, iy2) - pmat(0,iy1);
+	  trans(1,1) = pmat(1, iy2) - pmat(1,iy1);
+
+	  double det = trans(0,0)*trans(1,1)-trans(1,0)*trans(0,1);
+
+	  if (det <= 0)
+	    {
+	      dd = 0;
+	      return 1e12;
+	    }
+	  
+	  dtrans(0,0) = vmat(0, ix2) - vmat(0,ix1);
+	  dtrans(1,0) = vmat(1, ix2) - vmat(1,ix1);
+	  dtrans(0,1) = vmat(0, iy2) - vmat(0,iy1);
+	  dtrans(1,1) = vmat(1, iy2) - vmat(1,iy1);
+
+
+	  // Frobenius norm
+	  double frob = 0;
+	  for (int j = 0; j < 4; j++) 
+	    frob += sqr (trans(j));
+	  frob = sqrt (frob);
+	  
+	  double dfrob = 0;
+	  for (int j = 0; j < 4; j++)
+	    dfrob += trans(j) * dtrans(j);
+	  dfrob = dfrob / frob;
+	  
+	  frob /= 2;      
+	  dfrob /= 2;
+	  
+	  
+	  // ddet = \sum_j det (m_j)   with m_j = trans, except col j = dtrans
+	  double ddet 
+	    = dtrans(0,0) * trans(1,1) - trans(0,1) * dtrans(1,0)
+	    + trans(0,0) * dtrans(1,1) - dtrans(0,1) * trans(1,0);
+	  
+	  err += frob * frob / det;
+	  dd += (2 * frob * dfrob * det - frob * frob * ddet) / (det * det);
+	}
+      
+      err /= 4;
+      dd /= 4;
+      return err;
+    }
+
   int nip = GetNIP();
-  static DenseMatrix trans(2,2), dtrans(2,2), hmat(2,2);
+  static DenseMatrix trans(2,2), dtrans(2,2);
   static DenseMatrix pmat, vmat;
   
   pmat.SetSize (2, GetNP());
@@ -494,50 +618,39 @@ CalcJacobianBadnessDirDeriv (const ARRAY<Point2d> & points,
 
   GetPointMatrix (points, pmat);
   
-  for (i = 1; i <= np; i++)
-    for (j = 1; j <= 2; j++)
-      vmat.Elem(j, i) = 0;
-  //  for (j = 1; j <= 2; j++)
+  vmat = 0.0;
   vmat.Elem(1, pi) = dir.X();
   vmat.Elem(2, pi) = dir.Y();
 
 
-
   double err = 0;
   dd = 0;
 
-  for (i = 1; i <= nip; i++)
+  for (int i = 1; i <= nip; i++)
     {
       GetTransformation (i, pmat, trans);
       GetTransformation (i, vmat, dtrans);
 
-
       // Frobenius norm
       double frob = 0;
-      for (j = 1; j <= 4; j++)
+      for (int j = 1; j <= 4; j++)
 	frob += sqr (trans.Get(j));
       frob = sqrt (frob);
       
       double dfrob = 0;
-      for (j = 1; j <= 4; j++)
+      for (int j = 1; j <= 4; j++)
 	dfrob += trans.Get(j) * dtrans.Get(j);
       dfrob = dfrob / frob;
       
       frob /= 2;      
       dfrob /= 2;
-
-      
-      double det = trans.Det();
-      double ddet = 0;
       
-      for (j = 1; j <= 2; j++)
-	{
-	  hmat = trans;
-	  for (k = 1; k <= 2; k++)
-	    hmat.Elem(k, j) = dtrans.Get(k, j);
-	  ddet += hmat.Det();
-	}
+      double det = trans(0,0)*trans(1,1)-trans(1,0)*trans(0,1);
 
+      // ddet = \sum_j det (m_j)   with m_j = trans, except col j = dtrans
+      double ddet 
+	= dtrans(0,0) * trans(1,1) - trans(0,1) * dtrans(1,0)
+	+ trans(0,0) * dtrans(1,1) - dtrans(0,1) * trans(1,0);
 
       if (det <= 0)
 	err += 1e12;
@@ -609,8 +722,7 @@ void Element2d :: ComputeIntegrationPointData () const
     case 4: if (ipdquad.Size()) return; break;
     }
 
-  int i;
-  for (i = 1; i <= GetNIP(); i++)
+  for (int i = 1; i <= GetNIP(); i++)
     {
       IntegrationPointData * ipd = new IntegrationPointData;
       Point2d hp;
@@ -672,9 +784,11 @@ Element :: Element ()
   flags.reverse = 0;
   flags.illegal = 0;
   flags.illegal_valid = 0;
+  flags.badness_valid = 0;
   flags.refflag = 1;
   flags.deleted = 0;
   order = 1;
+  partitionNumber = -1;
 }
 
 
@@ -690,6 +804,7 @@ Element :: Element (int anp)
   flags.reverse = 0;
   flags.illegal = 0;
   flags.illegal_valid = 0;
+  flags.badness_valid = 0;
   flags.refflag = 1;
   flags.deleted = 0;
   switch (np)
@@ -718,6 +833,7 @@ Element :: Element (ELEMENT_TYPE type)
   flags.reverse = 0;
   flags.illegal = 0;
   flags.illegal_valid = 0;
+  flags.badness_valid = 0;
   flags.refflag = 1;
   flags.deleted = 0;
   order = 1;
@@ -729,14 +845,14 @@ Element :: Element (ELEMENT_TYPE type)
 
 Element & Element :: operator= (const Element & el2)
 {
-  int i;
   typ = el2.typ;
   np = el2.np;
-  for (i = 0; i < ELEMENT_MAXPOINTS; i++)
+  for (int i = 0; i < ELEMENT_MAXPOINTS; i++)
     pnum[i] = el2.pnum[i];
   index = el2.index;
   flags = el2.flags;
   order = el2.order;
+  hp_elnr = el2.hp_elnr;
   return *this;
 }
 
@@ -1434,36 +1550,130 @@ void Element :: GetShape (const Point3d & p, Vector & shape) const
   switch (typ)
     {
     case TET:
-      shape.Elem(1) = 1 - p.X() - p.Y() - p.Z(); 
-      shape.Elem(2) = p.X();
-      shape.Elem(3) = p.Y();
-      shape.Elem(4) = p.Z();
-      break;
+      {
+	shape.Elem(1) = 1 - p.X() - p.Y() - p.Z(); 
+	shape.Elem(2) = p.X();
+	shape.Elem(3) = p.Y();
+	shape.Elem(4) = p.Z();
+	break;
+      }
     case TET10:
-      double lam1 = 1 - p.X() - p.Y() - p.Z();
-      double lam2 = p.X();
-      double lam3 = p.Y();
-      double lam4 = p.Z();
-      
-      shape.Elem(5) = 4 * lam1 * lam2;
-      shape.Elem(6) = 4 * lam1 * lam3;
-      shape.Elem(7) = 4 * lam1 * lam4;
-      shape.Elem(8) = 4 * lam2 * lam3;
-      shape.Elem(9) = 4 * lam2 * lam4;
-      shape.Elem(10) = 4 * lam3 * lam4;
-
-      shape.Elem(1) = lam1 - 
-	0.5 * (shape.Elem(5) + shape.Elem(6) + shape.Elem(7));
-      shape.Elem(2) = lam2 - 
-	0.5 * (shape.Elem(5) + shape.Elem(8) + shape.Elem(9));
-      shape.Elem(3) = lam3 - 
-	0.5 * (shape.Elem(6) + shape.Elem(8) + shape.Elem(10));
-      shape.Elem(4) = lam4 - 
-	0.5 * (shape.Elem(7) + shape.Elem(9) + shape.Elem(10));
-      break;
+      {
+	double lam1 = 1 - p.X() - p.Y() - p.Z();
+	double lam2 = p.X();
+	double lam3 = p.Y();
+	double lam4 = p.Z();
+	
+	shape.Elem(5) = 4 * lam1 * lam2;
+	shape.Elem(6) = 4 * lam1 * lam3;
+	shape.Elem(7) = 4 * lam1 * lam4;
+	shape.Elem(8) = 4 * lam2 * lam3;
+	shape.Elem(9) = 4 * lam2 * lam4;
+	shape.Elem(10) = 4 * lam3 * lam4;
+	
+	shape.Elem(1) = lam1 - 
+	  0.5 * (shape.Elem(5) + shape.Elem(6) + shape.Elem(7));
+	shape.Elem(2) = lam2 - 
+	  0.5 * (shape.Elem(5) + shape.Elem(8) + shape.Elem(9));
+	shape.Elem(3) = lam3 - 
+	  0.5 * (shape.Elem(6) + shape.Elem(8) + shape.Elem(10));
+	shape.Elem(4) = lam4 - 
+	  0.5 * (shape.Elem(7) + shape.Elem(9) + shape.Elem(10));
+	break;
+      }
+
+    case PRISM:
+      {
+	Point<3> hp = p; 
+	shape(0) = hp(0) * (1-hp(2));
+	shape(1) = hp(1) * (1-hp(2));
+	shape(2) = (1-hp(0)-hp(1)) * (1-hp(2));
+	shape(3) = hp(0) * hp(2);
+	shape(4) = hp(1) * hp(2);
+	shape(5) = (1-hp(0)-hp(1)) * hp(2);
+	break;
+      }
+    case HEX:
+      {
+	Point<3> hp = p; 
+	shape(0) = (1-hp(0))*(1-hp(1))*(1-hp(2));
+	shape(1) = (  hp(0))*(1-hp(1))*(1-hp(2));
+	shape(2) = (  hp(0))*(  hp(1))*(1-hp(2));
+	shape(3) = (1-hp(0))*(  hp(1))*(1-hp(2));
+	shape(4) = (1-hp(0))*(1-hp(1))*(  hp(2));
+	shape(5) = (  hp(0))*(1-hp(1))*(  hp(2));
+	shape(6) = (  hp(0))*(  hp(1))*(  hp(2));
+	shape(7) = (1-hp(0))*(  hp(1))*(  hp(2));
+	break;
+      }
     }
 }
 
+
+void Element :: GetShapeNew (const Point<3> & p, FlatVector & shape) const
+{
+  /*
+  if (shape.Size() < GetNP())
+    {
+      cerr << "Element::GetShape: Length not fitting" << endl;
+      return;
+    }
+  */
+
+  switch (typ)
+    {
+    case TET:
+      {
+	shape(0) = p(0);
+	shape(1) = p(1);
+	shape(2) = p(2);
+	shape(3) = 1-p(0)-p(1)-p(2);
+	break;
+      }
+
+    case PYRAMID:
+      {
+	double noz = 1-p(2);
+	if (noz == 0.0) noz = 1e-10;
+
+	double xi  = p(0) / noz;
+	double eta = p(1) / noz;
+	shape(0) = (1-xi)*(1-eta) * (noz);
+	shape(1) = (  xi)*(1-eta) * (noz);
+	shape(2) = (  xi)*(  eta) * (noz);
+	shape(3) = (1-xi)*(  eta) * (noz);
+	shape(4) = p(2);
+	break;
+      }
+
+    case PRISM:
+      {
+	shape(0) = p(0) * (1-p(2));
+	shape(1) = p(1) * (1-p(2));
+	shape(2) = (1-p(0)-p(1)) * (1-p(2));
+	shape(3) = p(0) * p(2);
+	shape(4) = p(1) * p(2);
+	shape(5) = (1-p(0)-p(1)) * p(2);
+	break;
+      }
+    case HEX:
+      {
+	shape(0) = (1-p(0))*(1-p(1))*(1-p(2));
+	shape(1) = (  p(0))*(1-p(1))*(1-p(2));
+	shape(2) = (  p(0))*(  p(1))*(1-p(2));
+	shape(3) = (1-p(0))*(  p(1))*(1-p(2));
+	shape(4) = (1-p(0))*(1-p(1))*(  p(2));
+	shape(5) = (  p(0))*(1-p(1))*(  p(2));
+	shape(6) = (  p(0))*(  p(1))*(  p(2));
+	shape(7) = (1-p(0))*(  p(1))*(  p(2));
+	break;
+      }
+    }
+}
+
+
+
+
 void Element :: 
 GetDShape (const Point3d & p, DenseMatrix & dshape) const
 {
@@ -1492,19 +1702,78 @@ GetDShape (const Point3d & p, DenseMatrix & dshape) const
 }
 
 
+
+void Element :: 
+GetDShapeNew (const Point<3> & p, MatrixFixWidth<3> & dshape) const
+{
+  switch (typ)
+    {
+    case TET:
+      {
+	dshape = 0;
+	dshape(0,0) = 1;
+	dshape(1,1) = 1;
+	dshape(2,2) = 1;
+	dshape(3,0) = -1;
+	dshape(3,1) = -1;
+	dshape(3,2) = -1;
+	break;
+      }
+    case PRISM:
+      {
+	dshape = 0;
+	dshape(0,0) = 1-p(2);
+	dshape(0,2) = -p(0);
+	dshape(1,1) = 1-p(2);
+	dshape(1,2) = -p(1);
+	dshape(2,0) = -(1-p(2));
+	dshape(2,1) = -(1-p(2));
+	dshape(2,2) = -(1-p(0)-p(1));
+
+	dshape(3,0) = p(2);
+	dshape(3,2) = p(0);
+	dshape(4,1) = p(2);
+	dshape(4,2) = p(1);
+	dshape(5,0) = -p(2);
+	dshape(5,1) = -p(2);
+	dshape(5,2) = 1-p(0)-p(1);
+	break;
+      }
+
+    default:
+      {
+	int np = GetNP();
+	double eps = 1e-6;
+	Vector shaper(np), shapel(np);
+	
+	for (int i = 1; i <= 3; i++)
+	  {
+	    Point3d pr(p), pl(p);
+	    pr.X(i) += eps;
+	    pl.X(i) -= eps;
+	    
+	    GetShapeNew (pr, shaper);
+	    GetShapeNew (pl, shapel);
+	    for (int j = 1; j <= np; j++)
+	      dshape.Elem(j, i) = (shaper.Get(j) - shapel.Get(j)) / (2 * eps);
+	  }
+      }
+    }
+}
+
 void Element :: 
 GetPointMatrix (const T_POINTS & points,
 		DenseMatrix & pmat) const
 {
-  int i;
   int np = GetNP();
+  /*
   if (pmat.Width() != np || pmat.Height() != 3)
     {
       cerr << "Element::GetPointMatrix: sizes don't fit" << endl;
       return;
     }
-  
-  for (i = 1; i <= np; i++)
+  */
+  for (int i = 1; i <= np; i++)
     {
       const Point3d & p = points.Get(PNum(i));
       pmat.Elem(1, i) = p.X();
@@ -1672,8 +1941,8 @@ void Element :: ComputeIntegrationPointData () const
 
 FaceDescriptor ::  FaceDescriptor()
 { 
-  surfnr = domin = 
-    domout  = bcprop = 0; 
+  surfnr = domin = domout  = bcprop = 0; 
+  domin_singular = domout_singular = 0;
   tlosurf = -1;
 }
 
@@ -1685,6 +1954,7 @@ FaceDescriptor(int surfnri, int domini, int domouti, int tlosurfi)
   domout = domouti;
   tlosurf = tlosurfi; 
   bcprop = surfnri;
+  domin_singular = domout_singular = 0;
 }
 
 FaceDescriptor :: FaceDescriptor(const Segment & seg)
@@ -1694,6 +1964,7 @@ FaceDescriptor :: FaceDescriptor(const Segment & seg)
   domout = seg.domout+1;
   tlosurf = seg.tlosurf+1;
   bcprop = 0;
+  domin_singular = domout_singular = 0;
 }
 
 int FaceDescriptor ::  SegmentFits (const Segment & seg)
@@ -1712,7 +1983,9 @@ ostream & operator<<(ostream  & s, const FaceDescriptor & fd)
     << ", domin = " << fd.domin
     << ", domout = " << fd.domout
     << ", tlosurf = " << fd.tlosurf
-    << ", bcprop = " << fd.bcprop;
+    << ", bcprop = " << fd.bcprop
+    << ", domin_sing = " << fd.domin_singular
+    << ", domout_sing = " << fd.domout_singular;
   return s;
 }
 
@@ -1771,26 +2044,20 @@ int Identifications :: GetSymmetric (PointIndex pi1, PointIndex pi2) const
 }
 
 
-void Identifications :: GetMap (int identnr, 
-				ARRAY<int,PointIndex::BASE> & identmap) const
+void Identifications :: GetMap (int identnr, ARRAY<int,PointIndex::BASE> & identmap) const
 {
-  int i, j;
-  
   identmap.SetSize (mesh.GetNP());
-  for (i = 1; i <= identmap.Size(); i++)
-    identmap.Elem(i) = 0;
+  identmap = 0;
 
-  for (i = 1; i <= identifiedpoints->GetNBags(); i++)
-    for (j = 1; j <= identifiedpoints->GetBagSize(i); j++)
+  for (int i = 1; i <= identifiedpoints->GetNBags(); i++)
+    for (int j = 1; j <= identifiedpoints->GetBagSize(i); j++)
       {
 	INDEX_2 i2;
 	int nr;
 	identifiedpoints->GetData (i, j, i2, nr);
 	
-	if (nr == identnr)
-	  {
-	    identmap.Elem(i2.I1()) = i2.I2();
-	  }
+	if (nr == identnr || !identnr)
+	  identmap.Elem(i2.I1()) = i2.I2();
       }  
 }
  
@@ -1815,6 +2082,22 @@ void Identifications :: GetPairs (int identnr,
 }
 
 
+void Identifications :: SetMaxPointNr (int maxpnum)
+{
+  for (int i = 1; i <= identifiedpoints->GetNBags(); i++)
+    for (int j = 1; j <= identifiedpoints->GetBagSize(i); j++)
+      {
+	INDEX_2 i2;
+	int nr;
+	identifiedpoints->GetData (i, j, i2, nr);
+	
+	if (i2.I1() > maxpnum || i2.I2() > maxpnum)
+	  {
+	    i2.I1() = i2.I2() = -1;
+	    identifiedpoints->SetData (i, j, i2, -1);	    
+	  }
+      }
+}
 
 
 
@@ -1825,14 +2108,14 @@ MeshingParameters :: MeshingParameters ()
   optimize3d = "cmdmstm";
   optsteps3d = 3;
   optimize2d = "smsmsmSmSmSm";
-  optsteps2d = 1;
+  optsteps2d = 3;
   opterrpow = 2;
   blockfill = 1;
   filldist = 0.1;
   safety = 5;
   relinnersafety = 3;
   uselocalh = 1;
-  grading = 0.5;
+  grading = 0.3;
   delaunay = 1;
   maxh = 1e10;
   meshsizefilename = NULL;
@@ -1840,9 +2123,10 @@ MeshingParameters :: MeshingParameters ()
   checkoverlap = 1;
   checkchartboundary = 1;
   curvaturesafety = 2;
+  segmentsperedge = 1;
   parthread = 0;
-  
-  elsizeweight = 0;
+
+  elsizeweight = 0.2;
   giveuptol = 10;
   maxoutersteps = 5;
   starshapeclass = 5;
@@ -1856,13 +2140,81 @@ MeshingParameters :: MeshingParameters ()
 void MeshingParameters :: Print (ostream & ost) const
 {
   ost << "Meshing parameters: " << endl
-      << "maxh = " << maxh << endl
-      << "grading = " << grading << endl
-      << "startinsurface = " << startinsurface << endl
-      << "checkoverlap = " << checkoverlap << endl
-      << "curvaturesafety = " << curvaturesafety << endl;
+      << "optimize3d = " << optimize3d << endl
+      << "optsteps3d = " << optsteps3d << endl
+      << " optimize2d = " <<  optimize2d << endl
+      << " optsteps2d = " <<  optsteps2d << endl
+      << " opterrpow = " <<  opterrpow << endl
+      << " blockfill = " <<  blockfill << endl
+      << " filldist = " <<  filldist << endl
+      << " safety = " <<  safety << endl
+      << " relinnersafety = " <<  relinnersafety << endl
+      << " uselocalh = " <<  uselocalh << endl
+      << " grading = " <<  grading << endl
+      << " delaunay = " <<  delaunay << endl
+      << " maxh = " <<  maxh << endl;
+  if(meshsizefilename)
+    ost << " meshsizefilename = " <<  meshsizefilename << endl;
+  else
+    ost << " meshsizefilename = NULL" << endl;
+  ost << " startinsurface = " <<  startinsurface << endl
+      << " checkoverlap = " <<  checkoverlap << endl
+      << " checkchartboundary = " <<  checkchartboundary << endl
+      << " curvaturesafety = " <<  curvaturesafety << endl
+      << " segmentsperedge = " <<  segmentsperedge << endl
+      << " parthread = " <<  parthread << endl
+      << " elsizeweight = " <<  elsizeweight << endl
+      << " giveuptol = " <<  giveuptol << endl
+      << " maxoutersteps = " <<  maxoutersteps << endl
+      << " starshapeclass = " <<  starshapeclass << endl
+      << " baseelnp        = " <<  baseelnp        << endl
+      << " sloppy = " <<  sloppy << endl
+      << " badellimit = " <<  badellimit << endl
+      << " secondorder = " <<  secondorder << endl
+      << " elementorder = " <<  elementorder << endl
+      << " quad = " <<  quad << endl
+      << " inverttets = " <<  inverttets << endl
+      << " inverttrigs = " <<  inverttrigs << endl;
 }
 
+void MeshingParameters :: CopyFrom(const MeshingParameters & other)
+{
+  //strcpy(optimize3d,other.optimize3d); 
+  optimize3d = other.optimize3d;
+  optsteps3d = other.optsteps3d;
+  //strcpy(optimize2d,other.optimize2d); 
+  optimize2d = other.optimize2d;
+  optsteps2d = other.optsteps2d;
+  opterrpow = other.opterrpow;
+  blockfill = other.blockfill;
+  filldist = other.filldist;
+  safety = other.safety;
+  relinnersafety = other.relinnersafety;
+  uselocalh = other.uselocalh;
+  grading = other.grading;
+  delaunay = other.delaunay;
+  maxh = other.maxh;
+  //strcpy(const_cast<char*>(meshsizefilename), other.meshsizefilename);
+  //const_cast<char*>(meshsizefilename) = other.meshsizefilename; //???
+  startinsurface = other.startinsurface;
+  checkoverlap = other.checkoverlap;
+  checkchartboundary = other.checkchartboundary;
+  curvaturesafety = other.curvaturesafety;
+  segmentsperedge = other.segmentsperedge;
+  parthread = other.parthread;
+  elsizeweight = other.elsizeweight;
+  giveuptol = other.giveuptol;
+  maxoutersteps = other.maxoutersteps;
+  starshapeclass = other.starshapeclass;
+  baseelnp = other.baseelnp;       
+  sloppy = other.sloppy;
+  badellimit = other.badellimit;
+  secondorder = other.secondorder;
+  elementorder = other.elementorder;
+  quad = other.quad;
+  inverttets = other.inverttets;
+  inverttrigs = other.inverttrigs;
+}
 
 
 DebugParameters :: DebugParameters ()
@@ -1870,6 +2222,7 @@ DebugParameters :: DebugParameters ()
   slowchecks = 0;
   haltsuccess = 0;
   haltnosuccess = 0;
+  haltlargequalclass = 0;
   haltsegment = 0;
   haltsegmentp1 = 0;
   haltsegmentp2 = 0;
diff --git a/Netgen/libsrc/meshing/meshtype.hpp b/Netgen/libsrc/meshing/meshtype.hpp
index 25a131114a..56cc5f902d 100644
--- a/Netgen/libsrc/meshing/meshtype.hpp
+++ b/Netgen/libsrc/meshing/meshtype.hpp
@@ -1,6 +1,8 @@
 #ifndef MESHTYPE
 #define MESHTYPE
 
+//#include <algorithm>
+
 /**************************************************************************/
 /* File:   meshtype.hpp                                                   */
 /* Author: Joachim Schoeberl                                              */
@@ -25,7 +27,7 @@ typedef int ELEMENT_FACE[4];      // points, last one is -1 for trig
 
 
 #define ELEMENT_MAXPOINTS 12
-#define ELEMENT2D_MAXPOINTS 6
+#define ELEMENT2D_MAXPOINTS 8
 
 
 enum POINTTYPE { FIXEDPOINT = 1, EDGEPOINT = 2, SURFACEPOINT = 3, INNERPOINT = 4 };
@@ -40,10 +42,14 @@ enum OPTIMIZEGOAL { OPT_QUALITY, OPT_CONFORM, OPT_REST, OPT_WORSTCASE, OPT_LEGAL
 extern int GetTimeStamp();
 extern int NextTimeStamp();
 
-struct PointGeomInfo
+class PointGeomInfo
 {
+public:
   int trignum;   // for STL Meshing
   double u, v;   // for OCC Meshing
+
+  PointGeomInfo () 
+    : trignum(-1), u(0), v(0) { ; }
 };
 
 inline ostream & operator<< (ostream & ost, const PointGeomInfo & gi)
@@ -69,7 +75,7 @@ public:
 };
 
 
-struct EdgePointGeomInfo
+class EdgePointGeomInfo
 {
 public:
   int edgenr;
@@ -77,7 +83,14 @@ public:
   double u, v; // for OCC Meshing
 
   EdgePointGeomInfo ()
-  { edgenr = 0; dist = 0; u = v = 0; }
+    : edgenr(0), dist(0.0), u(0.0), v(0.0) { ; }
+
+  EdgePointGeomInfo & operator= (const EdgePointGeomInfo & gi2)
+  {
+    edgenr = gi2.edgenr;  dist = gi2.dist;
+    u = gi2.u; v = gi2.v;
+    return *this;
+  }
 };
 
 inline ostream & operator<< (ostream & ost, const EdgePointGeomInfo & gi)
@@ -89,7 +102,6 @@ inline ostream & operator<< (ostream & ost, const EdgePointGeomInfo & gi)
 
 
 
-
 class PointIndex
 {
   int i;
@@ -103,7 +115,11 @@ public:
   PointIndex operator++ (int) { int hi = i; i++; return hi; }
   PointIndex operator-- (int) { int hi = i; i--; return hi; }
 
+#ifdef BASE0
+  enum { BASE = 0 };
+#else
   enum { BASE = 1 };
+#endif  
 };
 
 inline istream & operator>> (istream & ist, PointIndex & pi)
@@ -132,7 +148,15 @@ public:
   ElementIndex & operator-- (int) { i--; return *this; }
 };
 
+inline istream & operator>> (istream & ist, ElementIndex & pi)
+{
+  int i; ist >> i; pi = i; return ist;
+}
 
+inline ostream & operator<< (ostream & ost, const ElementIndex & pi)
+{
+  return (ost << int(pi));
+}
 
 
 class SurfaceElementIndex
@@ -149,7 +173,15 @@ public:
   SurfaceElementIndex & operator-- (int) { i--; return *this; }
 };
 
+inline istream & operator>> (istream & ist, SurfaceElementIndex & pi)
+{
+  int i; ist >> i; pi = i; return ist;
+}
 
+inline ostream & operator<< (ostream & ost, const SurfaceElementIndex & pi)
+{
+  return (ost << int(pi));
+}
 
 class SegmentIndex
 {
@@ -165,7 +197,15 @@ public:
   SegmentIndex & operator-- (int) { i--; return *this; }
 };
 
+inline istream & operator>> (istream & ist, SegmentIndex & pi)
+{
+  int i; ist >> i; pi = i; return ist;
+}
 
+inline ostream & operator<< (ostream & ost, const SegmentIndex & pi)
+{
+  return (ost << int(pi));
+}
 
 
 
@@ -178,14 +218,22 @@ public:
 class MeshPoint : public Point3d
 {
   int layer;
+  bool singular;
+  POINTTYPE type;
 public:
-  MeshPoint () { ; }
-  MeshPoint (const Point3d & ap, int alayer = 1)
-    : Point3d (ap), layer(alayer) { ; }
+  MeshPoint () : layer(1), singular(0), type(INNERPOINT) { ; }
+  MeshPoint (const Point3d & ap, int alayer = 1, POINTTYPE apt = INNERPOINT)
+    : Point3d (ap), layer(alayer), singular(0), type(apt) { ; }
   
   void SetPoint (const Point3d & ap)
   { Point3d::operator= (ap); }
   int GetLayer() const { return layer; }
+
+  bool IsSingular() const { return singular; }
+  void SetSingular(bool s = 1) { singular = s; }
+
+  POINTTYPE Type() const { return type; }
+  void SetType(POINTTYPE at) { type = at; }
 };
 
 
@@ -212,13 +260,15 @@ class Element2d
   ///
   ELEMENT_TYPE typ:6;
   /// number of points
-  unsigned int np:3;
+  unsigned int np:4;
   bool badel:1;
   bool refflag:1;  // marked for refinement
   bool deleted:1;  // element is deleted
 
   /// order for hp-FEM
   unsigned int order:6;
+
+
 public:
   ///
   Element2d (int anp = 3);
@@ -231,7 +281,20 @@ public:
   ///
   ELEMENT_TYPE GetType () const { return typ; }
   /// 
-  void SetType (ELEMENT_TYPE atyp);
+  void SetType (ELEMENT_TYPE atyp)
+  {
+    typ = atyp;
+    switch (typ)
+      {
+      case TRIG: np = 3; break;
+      case QUAD: np = 4; break;
+      case TRIG6: np = 6; break;
+      case QUAD6: np = 6; break;
+      case QUAD8: np = 8; break;
+      default:
+	PrintSysError ("Element2d::SetType, illegal type ", typ);
+      }
+  }
   ///
   int GetNP() const { return np; }
   ///
@@ -339,7 +402,7 @@ public:
     return deleted; 
   }
 
-  void SetRefinementFlag (int rflag = 1) 
+  void SetRefinementFlag (bool rflag = 1) 
   { refflag = rflag; }
   bool TestRefinementFlag () const
   { return refflag; }
@@ -347,6 +410,8 @@ public:
   int HasFace(const Element2d& el) const;
   ///
   int meshdocval;
+  ///
+  int hp_elnr;
 };
 
 
@@ -382,7 +447,8 @@ private:
   /// number of points (4..tet, 5..pyramid, 6..prism, 8..hex, 10..quad tet, 12..quad prism)
   int np:5;
   ///
-  struct flagstruct {
+  class flagstruct { 
+  public:
     bool marked:1;  // marked for refinement
     bool badel:1;   // angles worse then limit
     bool reverse:1; // for refinement a la Bey
@@ -398,6 +464,9 @@ private:
   unsigned int order:6;
   /// stored shape-badness of element
   float badness;
+  /// number of partition for parallel compution 
+  short int partitionNumber;
+  ///
   
 public:
   flagstruct flags;
@@ -425,9 +494,10 @@ public:
       case TET: return 4;
       case TET10: return 4;
       case PRISM12: return 6;
+      case PRISM: return 6; //SZ 
       default:
 #ifdef DEBUG
-	PrintSysError ("Element3d::GetNV not implemented for typ", typ)
+	PrintSysError ("Element3d::GetNV not implemented for typ ", typ)
 #endif
 	  ;
       }
@@ -513,8 +583,10 @@ public:
 			  class DenseMatrix & trans) const;
 
   void GetShape (const Point3d & p, class Vector & shape) const;
+  void GetShapeNew (const Point<3> & p, class FlatVector & shape) const;
   /// matrix 2 * np
   void GetDShape (const Point3d & p, class DenseMatrix & dshape) const;
+  void GetDShapeNew (const Point<3> & p, class MatrixFixWidth<3> & dshape) const;
   /// matrix 3 * np
   void GetPointMatrix (const T_POINTS & points,
 		       class DenseMatrix & pmat) const; 
@@ -529,7 +601,7 @@ public:
   ///
   friend ostream & operator<<(ostream  & s, const Element & el);
 
-  void SetRefinementFlag (int rflag = 1) 
+  void SetRefinementFlag (bool rflag = 1) 
   { flags.refflag = rflag; }
   int TestRefinementFlag () const
   { return flags.refflag; }
@@ -559,6 +631,11 @@ public:
 
     return flags.deleted; 
   }
+
+  int GetPartition () const { return partitionNumber; }
+  void SetPartition (int nr) { partitionNumber = nr; }; 
+
+  int hp_elnr;
 };
 
 
@@ -584,9 +661,12 @@ public:
   /// edge nr
   int edgenr;
   ///
-  unsigned int singedge:1;
+  unsigned int singedge_left:1;
+  unsigned int singedge_right:1;
+
   /// 0.. not first segment of segs, 1..first of class, 2..first of class, inverse
   unsigned int seginfo:2;
+
   /// surface decoding index
   int si;          
   /// domain number inner side
@@ -597,6 +677,7 @@ public:
   int tlosurf;
   ///
   PointGeomInfo geominfo[2];
+
   /// surfaces describing edge
   int surfnr1, surfnr2;
   ///
@@ -605,6 +686,18 @@ public:
   int pmid; // for second order
   ///
   int meshdocval;
+
+
+  PointIndex operator[] (int i) const
+  { return (i == 0) ? p1 : p2; }
+
+  PointIndex & operator[] (int i) 
+  { return (i == 0) ? p1 : p2; }
+
+  Segment& operator=(const Segment & other);
+
+  
+  int hp_elnr;
 };
 
 
@@ -644,6 +737,13 @@ public:
   void SetBCProperty (int bc) { bcprop = bc; }
 
   friend ostream & operator<<(ostream  & s, const FaceDescriptor & fd);
+
+
+  ///
+  bool domin_singular;
+  bool domout_singular;
+
+
 };
 
  
@@ -683,7 +783,7 @@ public:
   int optsteps2d;
   /// power of error (to approximate max err optimization)
   double opterrpow;
-  /// do block filling ?
+  /// do block filling ?  
   int blockfill;
   /// block filling up to distance
   double filldist;
@@ -747,6 +847,8 @@ public:
   MeshingParameters ();
   ///
   void Print (ostream & ost) const;
+
+  void CopyFrom(const MeshingParameters & other);
 };
 
 class DebugParameters 
@@ -761,6 +863,8 @@ public:
   ///
   int haltnosuccess;
   ///
+  int haltlargequalclass;
+  ///
   int haltsegment;
   ///
   int haltnode;
@@ -908,6 +1012,8 @@ public:
   ///
   int GetMaxNr () const { return maxidentnr; }  
 
+  /// remove secondorder
+  void SetMaxPointNr (int maxpnum);
 };
 
 
diff --git a/Netgen/libsrc/meshing/msghandler.cpp b/Netgen/libsrc/meshing/msghandler.cpp
index 8d2bb429ee..04da502a73 100644
--- a/Netgen/libsrc/meshing/msghandler.cpp
+++ b/Netgen/libsrc/meshing/msghandler.cpp
@@ -25,6 +25,24 @@ void PrintDot(char ch)
     }
 }
 
+void PrintMessage(int importance, 
+		  const MyStr& s1, const MyStr& s2)
+{
+  if (importance <= printmessage_importance)
+    {
+      Ng_PrintDest(MyStr(" ")+s1+s2+MyStr("\n"));
+    }
+}
+
+void PrintMessage(int importance, 
+		  const MyStr& s1, const MyStr& s2, const MyStr& s3, const MyStr& s4)
+{
+  if (importance <= printmessage_importance)
+    {
+      Ng_PrintDest(MyStr(" ")+s1+s2+s3+s4+MyStr("\n"));
+    }
+}
+
 void PrintMessage(int importance, 
 		  const MyStr& s1, const MyStr& s2, const MyStr& s3, const MyStr& s4, 
 		  const MyStr& s5, const MyStr& s6, const MyStr& s7, const MyStr& s8)
@@ -93,52 +111,35 @@ void PrintTime(const MyStr& s1, const MyStr& s2, const MyStr& s3, const MyStr& s
     Ng_PrintDest(MyStr(" Time = ")+s1+s2+s3+s4+s5+s6+s7+s8+MyStr("\n"));
 }
 
-ARRAY<MyStr*> msgstatus_stack(0);
-MyStr* msgstatus = NULL;
+
+static ARRAY<MyStr*> msgstatus_stack(0);
+static MyStr msgstatus = "";
+
+
+
 
 void ResetStatus()
 {
   SetStatMsg("idle");
 
-  if (msgstatus != NULL)
-    {
-      delete msgstatus;
-    } 
-  int i;
-  for (i = 1; i <= msgstatus_stack.Size(); i++)
-    {
-      delete msgstatus_stack.Get(i);
-    }
-  
+  for (int i = 0; i < msgstatus_stack.Size(); i++)
+    delete msgstatus_stack[i];
   msgstatus_stack.SetSize(0);
-  msgstatus = NULL;
-  multithread.task = "";
+
+  // multithread.task = "";
   multithread.percent = 100.;
 }
 
 void PushStatus(const MyStr& s)
 {
-  if (msgstatus == NULL)
-    {
-      SetStatMsg("idle");
-    } 
-  msgstatus_stack.Append(msgstatus);
-  msgstatus = NULL;
-
+  msgstatus_stack.Append(new MyStr (s));
   SetStatMsg(s);
-  //  multithread.task = "";
 }
+
 void PushStatusF(const MyStr& s)
 {
-  if (msgstatus == NULL)
-    {
-      SetStatMsg("idle");
-    } 
-  msgstatus_stack.Append(msgstatus);
-  msgstatus = NULL;
-
+  msgstatus_stack.Append(new MyStr (s));
   SetStatMsg(s);
-  //  multithread.task = "";
   PrintFnStart(s);
 }
 
@@ -147,13 +148,12 @@ void PopStatus()
   SetThreadPercent(100.);
   if (msgstatus_stack.Size())
     {
-      if (msgstatus != NULL) 
-	{
-	  delete msgstatus;
-	}
-      msgstatus = msgstatus_stack.Get(msgstatus_stack.Size());
+      if (msgstatus_stack.Size() > 1)
+	SetStatMsg (*msgstatus_stack.Last());
+      else
+	SetStatMsg ("");
+      delete msgstatus_stack.Last();
       msgstatus_stack.SetSize(msgstatus_stack.Size()-1);
-      multithread.task = msgstatus->c_str();
     }
   else
     {
@@ -167,14 +167,11 @@ void SetStatMsgF(const MyStr& s)
   SetStatMsg(s);
 }
 */
+
 void SetStatMsg(const MyStr& s)
 {
-  if (msgstatus != NULL)
-    {
-      delete msgstatus;
-    }
-  msgstatus = new MyStr(s);
-  multithread.task = msgstatus->c_str();  
+  msgstatus = s;
+  multithread.task = msgstatus.c_str();  
 }
 
 void SetThreadPercent(double percent)
@@ -183,4 +180,14 @@ void SetThreadPercent(double percent)
 }
 
 
+
+#ifdef SMALLLIB
+void Ng_PrintDest(const char * s){cout << s <<flush;}
+double GetTime(){return 0;}
+void MyError(const char * ch)
+{
+  cerr << ch << endl;
+}
+#endif
+
 }
diff --git a/Netgen/libsrc/meshing/msghandler.hpp b/Netgen/libsrc/meshing/msghandler.hpp
index b7d4481866..7de9425193 100644
--- a/Netgen/libsrc/meshing/msghandler.hpp
+++ b/Netgen/libsrc/meshing/msghandler.hpp
@@ -15,8 +15,13 @@ extern void PrintDot(char ch = '.');
 
 //importance: importance of message: 1=very important, 3=middle, 5=low, 7=unimportant
 extern void PrintMessage(int importance, 
-			 const MyStr& s1, const MyStr& s2="", const MyStr& s3="", const MyStr& s4="", 
-			 const MyStr& s5="", const MyStr& s6="", const MyStr& s7="", const MyStr& s8="");
+			 const MyStr& s1, const MyStr& s2=MyStr());
+extern void PrintMessage(int importance, 
+			 const MyStr& s1, const MyStr& s2, const MyStr& s3, const MyStr& s4=MyStr());
+extern void PrintMessage(int importance, 
+			 const MyStr& s1, const MyStr& s2, const MyStr& s3, const MyStr& s4, 
+			 const MyStr& s5, const MyStr& s6=MyStr(), const MyStr& s7=MyStr(), const MyStr& s8=MyStr());
+
 // CR without line-feed
 extern void PrintMessageCR(int importance, 
 			   const MyStr& s1, const MyStr& s2="", const MyStr& s3="", const MyStr& s4="", 
diff --git a/Netgen/libsrc/meshing/netrule2.cpp b/Netgen/libsrc/meshing/netrule2.cpp
index c0acd4938d..c75c69d521 100644
--- a/Netgen/libsrc/meshing/netrule2.cpp
+++ b/Netgen/libsrc/meshing/netrule2.cpp
@@ -13,7 +13,9 @@ netrule :: netrule ()
 
 netrule ::  ~netrule()
 {
-  delete name;
+  if(name != NULL) delete [] name;
+  for(int i=0; i<oldutofreearea_i.Size(); i++)
+    delete oldutofreearea_i[i];
 }
 
 
@@ -31,9 +33,8 @@ void netrule :: GetFreeArea (ARRAY<Point2d> & afreearea)
 
 void netrule :: SetFreeZoneTransformation (const Vector & devp, int tolclass)
 {
-  int i;
   double lam1 = 1.0/tolclass;
-  double lam2 = 1-lam1;
+  double lam2 = 1.-lam1;
 
   double mem1[100], mem2[100], mem3[100];
 
@@ -42,14 +43,20 @@ void netrule :: SetFreeZoneTransformation (const Vector & devp, int tolclass)
   FlatVector devfree1(vs, mem2);
   FlatVector devfree2(vs, mem3);
 
-  oldutofreearea.Mult (devp, devfree1);
-  oldutofreearealimit.Mult (devp, devfree2);
-  devfree.Set2 (lam1, devfree1, lam2, devfree2);
-
-  transfreezone.SetSize (freezone.Size());
-
+  if (tolclass <= oldutofreearea_i.Size())
+    {
+      oldutofreearea_i[tolclass-1] -> Mult (devp, devfree);
+    }
+  else
+    {
+      oldutofreearea.Mult (devp, devfree1);
+      oldutofreearealimit.Mult (devp, devfree2);
+      devfree.Set2 (lam1, devfree1, lam2, devfree2);
+    }
 
-  int fzs = transfreezone.Size();
+  
+  int fzs = freezone.Size();
+  transfreezone.SetSize (fzs);
 
   if (fzs > 0)
     {
@@ -59,7 +66,7 @@ void netrule :: SetFreeZoneTransformation (const Vector & devp, int tolclass)
       fzmaxy = fzminy = transfreezone[0].Y();
     }
 
-  for (i = 1; i < fzs; i++)
+  for (int i = 1; i < fzs; i++)
     {
       transfreezone[i].X() = lam1 * freezone[i].X() + lam2 * freezonelimit[i].X() + devfree[2*i];
       transfreezone[i].Y() = lam1 * freezone[i].Y() + lam2 * freezonelimit[i].Y() + devfree[2*i+1];
@@ -70,12 +77,13 @@ void netrule :: SetFreeZoneTransformation (const Vector & devp, int tolclass)
       if (transfreezone[i].Y() < fzminy) fzminy = transfreezone[i].Y();
     }
 
-  for (i = 0; i < fzs; i++)
+  for (int i = 0; i < fzs; i++)
     {
       Point2d p1 = transfreezone[i];
       Point2d p2 = transfreezone[(i+1) % fzs];
 
       Vec2d vn (p2.Y() - p1.Y(), p1.X() - p2.X());
+
       double len2 = vn.Length2();
 
       if (len2 < 1e-10)
@@ -86,27 +94,34 @@ void netrule :: SetFreeZoneTransformation (const Vector & devp, int tolclass)
 	}
       else
 	{
-	  vn *= 1/sqrt (len2);
+	  vn /= sqrt (len2);    // should not be necessary
 
 	  freesetinequ(i,0) = vn.X(); 
 	  freesetinequ(i,1) = vn.Y(); 
 	  freesetinequ(i,2) = -(p1.X() * vn.X() + p1.Y() * vn.Y());
 	}
+
+      /*
+      freesetinequ(i,0) = vn.X(); 
+      freesetinequ(i,1) = vn.Y(); 
+      freesetinequ(i,2) = -(p1.X() * vn.X() + p1.Y() * vn.Y());
+      */
     }
 }
 
 
+/*
 int netrule :: IsInFreeZone2 (const Point2d & p) const
 {
-  int i;
-
-  for (i = 1; i <= transfreezone.Size(); i++)
+  for (int i = 0; i < transfreezone.Size(); i++)
     {
-      if (freesetinequ.Get(i, 1) * p.X() + freesetinequ.Get(i, 2) * p.Y() +
-	  freesetinequ.Get(i, 3) > 0) return 0;
+      if (freesetinequ(i, 0) * p.X() + 
+	  freesetinequ(i, 1) * p.Y() +
+	  freesetinequ(i, 2) > 0) return 0;
     }
   return 1;
 }
+*/
 
 int netrule :: IsLineInFreeZone2 (const Point2d & p1, const Point2d & p2) const
 {
@@ -122,9 +137,9 @@ int netrule :: IsLineInFreeZone2 (const Point2d & p1, const Point2d & p2) const
   for (i = 1; i <= transfreezone.Size(); i++)
     {
       if (freesetinequ.Get(i, 1) * p1.X() + freesetinequ.Get(i, 2) * p1.Y() +
-	  freesetinequ.Get(i, 3) > -1e-5 &&
+	  freesetinequ.Get(i, 3) > -1e-6 &&
 	  freesetinequ.Get(i, 1) * p2.X() + freesetinequ.Get(i, 2) * p2.Y() +
-	  freesetinequ.Get(i, 3) > -1e-5
+	  freesetinequ.Get(i, 3) > -1e-6
 	  ) return 0;
     }
 
@@ -156,9 +171,8 @@ int netrule :: IsLineInFreeZone2 (const Point2d & p1, const Point2d & p2) const
 
 int netrule :: ConvexFreeZone () const
 {
-  int i, n;
-  n = transfreezone.Size();
-  for (i = 1; i <= n; i++)
+  int n = transfreezone.Size();
+  for (int i = 1; i <= n; i++)
     {
       if (! CCW (transfreezone.Get(i), 
 		 transfreezone.Get(i % n + 1),
@@ -169,7 +183,7 @@ int netrule :: ConvexFreeZone () const
 }
 
 
-
+/*
 float netrule :: CalcPointDist (int pi, const Point2d & p) const
 {
   float dx = p.X() - points.Get(pi).X();
@@ -178,7 +192,7 @@ float netrule :: CalcPointDist (int pi, const Point2d & p) const
 
   return tf->f1 * dx * dx + tf->f2 * dx * dy + tf->f3 * dy * dy;
 }
-
+*/
 
 float netrule :: CalcLineError (int li, const Vec2d & v) const
 {
diff --git a/Netgen/libsrc/meshing/netrule3.cpp b/Netgen/libsrc/meshing/netrule3.cpp
index b6c5967058..fe6a7417c1 100644
--- a/Netgen/libsrc/meshing/netrule3.cpp
+++ b/Netgen/libsrc/meshing/netrule3.cpp
@@ -17,16 +17,14 @@ vnetrule :: vnetrule ()
 
 vnetrule :: ~vnetrule ()
 {
-  int i;
-  if (strlen(name))
-    delete name;
-  for (i = 1; i <= freefaces.Size(); i++)
+  if (strlen(name)) delete [] name;
+  for (int i = 1; i <= freefaces.Size(); i++)
     delete freefaces.Elem(i);
-  for (i = 1; i <= freesets.Size(); i++)
+  for (int i = 1; i <= freesets.Size(); i++)
     delete freesets.Elem(i);
-  for (i = 1; i <= freeedges.Size(); i++)
+  for (int i = 1; i <= freeedges.Size(); i++)
     delete freeedges.Elem(i);
-  for (i = 1; i <= freefaceinequ.Size(); i++)
+  for (int i = 1; i <= freefaceinequ.Size(); i++)
     delete freefaceinequ.Elem(i);
   delete oldutofreezone;
   delete oldutofreezonelimit;
@@ -34,8 +32,7 @@ vnetrule :: ~vnetrule ()
 
 int vnetrule :: TestFlag (char flag) const
 {
-  int i;
-  for (i = 1; i <= flags.Size(); i++)
+  for (int i = 1; i <= flags.Size(); i++)
     if (flags.Get(i) == flag) return 1;
   return 0;
 }
@@ -52,8 +49,6 @@ void vnetrule :: SetFreeZoneTransformation (const Vector & allp, int tolclass)
   double lam1 = 1.0/(2 * tolclass - 1);
   double lam2 = 1-lam1;
 
-  //  MARK (setfz1);
-
   transfreezone.SetSize (freezone.Size());
   
   int np = points.Size();
diff --git a/Netgen/libsrc/meshing/parser2.cpp b/Netgen/libsrc/meshing/parser2.cpp
index 402207c259..48ef280eb4 100644
--- a/Netgen/libsrc/meshing/parser2.cpp
+++ b/Netgen/libsrc/meshing/parser2.cpp
@@ -53,7 +53,7 @@ void netrule :: LoadRule (istream & ist)
   ist.get (buf, sizeof(buf), '"');
   ist.get (ch);
 
-  delete name;
+  if(name != NULL) delete [] name;
   name = new char[strlen (buf) + 1];
   strcpy (name, buf);
   //  (*mycout) << "Rule " << name << " found." << endl;
@@ -83,9 +83,9 @@ void netrule :: LoadRule (istream & ist)
 	      noldp++;
 
 	      tolerances.SetSize (noldp);
-	      tolerances.Elem(noldp).f1 = 0;
+	      tolerances.Elem(noldp).f1 = 1.0;
 	      tolerances.Elem(noldp).f2 = 0;
-	      tolerances.Elem(noldp).f3 = 0;
+	      tolerances.Elem(noldp).f3 = 1.0;
 
 	      ist >> ch;
 	      while (ch != ';')
@@ -449,6 +449,15 @@ void netrule :: LoadRule (istream & ist)
       }
   }
 
+  oldutofreearea_i.SetSize (10);
+  for (i = 0; i < oldutofreearea_i.Size(); i++)
+    {
+      oldutofreearea_i[i] = new DenseMatrix (oldutofreearea.Height(), oldutofreearea.Width());
+      DenseMatrix & mati = *oldutofreearea_i[i];
+      for (int j = 0; j < oldutofreearea.Height(); j++)
+	for (int k = 0; k < oldutofreearea.Width(); k++)
+	  mati(j,k) = 1.0 / (i+1) * oldutofreearea(j,k) + (1 - 1.0/(i+1)) * oldutofreearealimit(j,k);
+    }
 }
 
 
diff --git a/Netgen/libsrc/meshing/parser3.cpp b/Netgen/libsrc/meshing/parser3.cpp
index 7bf6e513ac..23ee84802e 100644
--- a/Netgen/libsrc/meshing/parser3.cpp
+++ b/Netgen/libsrc/meshing/parser3.cpp
@@ -982,6 +982,6 @@ void Meshing3 :: LoadRules (const char * filename, const char ** prules)
 	}
     }
   delete ist;
-  delete tr1;
+  delete [] tr1;
 }
 }
diff --git a/Netgen/libsrc/meshing/prism2rls.cpp b/Netgen/libsrc/meshing/prism2rls.cpp
index baf0bef388..7e696554c0 100644
--- a/Netgen/libsrc/meshing/prism2rls.cpp
+++ b/Netgen/libsrc/meshing/prism2rls.cpp
@@ -53,6 +53,11 @@ const char * prismrules2[] = {
 "{ 0.25 P1, 0 P2, 0 P3, 0.25 P4, 0.25 P5, 0.25 P6 };\n",\
 "{ 0 P1, 0.25 P2, 0.25 P3, 0 P4, 0.25 P5, 0.25 P6 };\n",\
 "\n",\
+"freeset\n",\
+"1 2 4 5 6 7;\n",\
+"\n",\
+"freeset\n",\
+"2 3 4 5 6 8;\n",\
 "\n",\
 "endrule\n",\
 "\n",\
@@ -110,6 +115,12 @@ const char * prismrules2[] = {
 "{ 0.25 P1, 0 P2, 0 P3, 0.25 P4, 0.25 P5, 0.25 P6 };\n",\
 "{ 0 P1, 0.25 P2, 0.25 P3, 0 P4, 0.25 P5, 0.25 P6 };\n",\
 "\n",\
+"freeset\n",\
+"1 2 4 5 6 7;\n",\
+"\n",\
+"freeset\n",\
+"2 3 4 5 6 8;\n",\
+"\n",\
 "\n",\
 "endrule\n",\
 "\n",\
@@ -405,7 +416,7 @@ const char * prismrules2[] = {
 "\n",\
 "rule \"flat prism\"\n",\
 "\n",\
-"quality 1\n",\
+"quality 100\n",\
 "\n",\
 "mappoints\n",\
 "(0, 0, 0);\n",\
diff --git a/Netgen/libsrc/meshing/prism2rls_2.cpp b/Netgen/libsrc/meshing/prism2rls_2.cpp
new file mode 100644
index 0000000000..baf0bef388
--- /dev/null
+++ b/Netgen/libsrc/meshing/prism2rls_2.cpp
@@ -0,0 +1,446 @@
+namespace netgen
+{
+const char * prismrules2[] = {
+"tolfak 0.5\n",\
+"\n",\
+"rule \"prism on quad\"\n",\
+"\n",\
+"quality 1\n",\
+"\n",\
+"mappoints\n",\
+"(0, 0, 0);\n",\
+"(1, 0, 0);\n",\
+"(1, 1, 0);\n",\
+"(0, 1, 0);\n",\
+"(0.5, 0, -0.86);\n",\
+"(0.5, 1, -0.86);\n",\
+"\n",\
+"mapfaces\n",\
+"(1, 2, 3, 4) del;\n",\
+"(1, 5, 2) del;\n",\
+"(4, 3, 6) del;\n",\
+"\n",\
+"newpoints\n",\
+"\n",\
+"newfaces\n",\
+"(5, 2, 3, 6);\n",\
+"(1, 5, 6, 4);\n",\
+"\n",\
+"elements\n",\
+"(1, 5, 2, 4, 6, 3);\n",\
+"\n",\
+"orientations\n",\
+"(1, 2, 3, 5);\n",\
+"(1, 3, 4, 6);\n",\
+"\n",\
+"freezone2\n",\
+"{ 1 P1 };\n",\
+"{ 1 P2 };\n",\
+"{ 1 P3 };\n",\
+"{ 1 P4 };\n",\
+"{ 1 P5 };\n",\
+"{ 1 P6 };\n",\
+"{ 0.3 P1, -0.1 P2, -0.1 P3, 0.3 P4, 0.3 P5, 0.3 P6 };\n",\
+"{ -0.1 P1, 0.3 P2, 0.3 P3, -0.1 P4, 0.3 P5, 0.3 P6 };\n",\
+"\n",\
+"freezonelimit\n",\
+"{ 1 P1 };\n",\
+"{ 1 P2 };\n",\
+"{ 1 P3 };\n",\
+"{ 1 P4 };\n",\
+"{ 1 P5 };\n",\
+"{ 1 P6 };\n",\
+"{ 0.25 P1, 0 P2, 0 P3, 0.25 P4, 0.25 P5, 0.25 P6 };\n",\
+"{ 0 P1, 0.25 P2, 0.25 P3, 0 P4, 0.25 P5, 0.25 P6 };\n",\
+"\n",\
+"\n",\
+"endrule\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"rule \"prism on quad, one trig\"\n",\
+"\n",\
+"quality 2\n",\
+"\n",\
+"mappoints\n",\
+"(0, 0, 0);\n",\
+"(1, 0, 0);\n",\
+"(1, 1, 0);\n",\
+"(0, 1, 0);\n",\
+"(0.5, 0, -0.86);\n",\
+"(0.5, 1, -0.86);\n",\
+"\n",\
+"mapfaces\n",\
+"(1, 2, 3, 4) del;\n",\
+"(1, 5, 2) del;\n",\
+"\n",\
+"newpoints\n",\
+"\n",\
+"newfaces\n",\
+"(5, 2, 3, 6);\n",\
+"(1, 5, 6, 4);\n",\
+"(4, 6, 3);\n",\
+"\n",\
+"elements\n",\
+"(1, 5, 2, 4, 6, 3);\n",\
+"\n",\
+"orientations\n",\
+"(1, 2, 3, 5);\n",\
+"(1, 3, 4, 6);\n",\
+"\n",\
+"freezone2\n",\
+"{ 1 P1 };\n",\
+"{ 1 P2 };\n",\
+"{ 1 P3 };\n",\
+"{ 1 P4 };\n",\
+"{ 1 P5 };\n",\
+"{ 1 P6 };\n",\
+"{ 0.3 P1, -0.1 P2, -0.1 P3, 0.3 P4, 0.3 P5, 0.3 P6 };\n",\
+"{ -0.1 P1, 0.3 P2, 0.3 P3, -0.1 P4, 0.3 P5, 0.3 P6 };\n",\
+"\n",\
+"freezonelimit\n",\
+"{ 1 P1 };\n",\
+"{ 1 P2 };\n",\
+"{ 1 P3 };\n",\
+"{ 1 P4 };\n",\
+"{ 1 P5 };\n",\
+"{ 1 P6 };\n",\
+"{ 0.25 P1, 0 P2, 0 P3, 0.25 P4, 0.25 P5, 0.25 P6 };\n",\
+"{ 0 P1, 0.25 P2, 0.25 P3, 0 P4, 0.25 P5, 0.25 P6 };\n",\
+"\n",\
+"\n",\
+"endrule\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"rule \"prism on 2 quad\"\n",\
+"\n",\
+"quality 1\n",\
+"\n",\
+"mappoints\n",\
+"(0, 0, 0);\n",\
+"(1, 0, 0);\n",\
+"(1, 1, 0);\n",\
+"(0, 1, 0);\n",\
+"(0.5, 0, -0.86);\n",\
+"(0.5, 1, -0.86);\n",\
+"\n",\
+"mapfaces\n",\
+"(1, 2, 3, 4) del;\n",\
+"(2, 5, 6, 3) del;\n",\
+"(1, 5, 2) del;\n",\
+"(4, 3, 6) del;\n",\
+"\n",\
+"newpoints\n",\
+"\n",\
+"newfaces\n",\
+"(1, 5, 6, 4);\n",\
+"\n",\
+"elements\n",\
+"(1, 5, 2, 4, 6, 3);\n",\
+"\n",\
+"freezone2\n",\
+"{ 1 P1 };\n",\
+"{ 1 P2 };\n",\
+"{ 1 P3 };\n",\
+"{ 1 P4 };\n",\
+"{ 1 P5 };\n",\
+"{ 1 P6 };\n",\
+"{ 0.3 P1, -0.1 P2, -0.1 P3, 0.3 P4, 0.3 P5, 0.3 P6 };\n",\
+"\n",\
+"freezonelimit\n",\
+"{ 1 P1 };\n",\
+"{ 1 P2 };\n",\
+"{ 1 P3 };\n",\
+"{ 1 P4 };\n",\
+"{ 1 P5 };\n",\
+"{ 1 P6 };\n",\
+"{ 0.25 P1, 0 P2, 0 P3, 0.25 P4, 0.25 P5, 0.25 P6 };\n",\
+"\n",\
+"freeset\n",\
+"1 2 4 5 6 7;\n",\
+"\n",\
+"freeset\n",\
+"2 3 4 6;\n",\
+"\n",\
+"\n",\
+"endrule\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"rule \"prism on 2 quad, one trig\"\n",\
+"\n",\
+"quality 2\n",\
+"\n",\
+"mappoints\n",\
+"(0, 0, 0);\n",\
+"(1, 0, 0);\n",\
+"(1, 1, 0);\n",\
+"(0, 1, 0);\n",\
+"(0.5, 0, -0.86);\n",\
+"(0.5, 1, -0.86);\n",\
+"\n",\
+"mapfaces\n",\
+"(1, 2, 3, 4) del;\n",\
+"(2, 5, 6, 3) del;\n",\
+"(1, 5, 2) del;\n",\
+"\n",\
+"newpoints\n",\
+"\n",\
+"newfaces\n",\
+"(1, 5, 6, 4);\n",\
+"(4, 6, 3);\n",\
+"\n",\
+"elements\n",\
+"(1, 5, 2, 4, 6, 3);\n",\
+"\n",\
+"freezone2\n",\
+"{ 1 P1 };\n",\
+"{ 1 P2 };\n",\
+"{ 1 P3 };\n",\
+"{ 1 P4 };\n",\
+"{ 1 P5 };\n",\
+"{ 1 P6 };\n",\
+"{ 0.3 P1, -0.1 P2, -0.1 P3, 0.3 P4, 0.3 P5, 0.3 P6 };\n",\
+"\n",\
+"freezonelimit\n",\
+"{ 1 P1 };\n",\
+"{ 1 P2 };\n",\
+"{ 1 P3 };\n",\
+"{ 1 P4 };\n",\
+"{ 1 P5 };\n",\
+"{ 1 P6 };\n",\
+"{ 0.25 P1, 0 P2, 0 P3, 0.25 P4, 0.25 P5, 0.25 P6 };\n",\
+"\n",\
+"freeset\n",\
+"1 2 4 5 6 7;\n",\
+"\n",\
+"freeset\n",\
+"2 3 4 6;\n",\
+"\n",\
+"\n",\
+"endrule\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"rule \"prism on 2 quada\"\n",\
+"\n",\
+"quality 1\n",\
+"\n",\
+"mappoints\n",\
+"(0, 0, 0);\n",\
+"(1, 0, 0);\n",\
+"(1, 1, 0);\n",\
+"(0, 1, 0);\n",\
+"(0.5, 0, -0.86);\n",\
+"(0.5, 1, -0.86);\n",\
+"\n",\
+"mapfaces\n",\
+"(1, 2, 3, 4) del;\n",\
+"(5, 1, 4, 6) del;\n",\
+"(1, 5, 2) del;\n",\
+"(4, 3, 6) del;\n",\
+"\n",\
+"newpoints\n",\
+"\n",\
+"newfaces\n",\
+"(5, 2, 3, 6);\n",\
+"\n",\
+"elements\n",\
+"(1, 5, 2, 4, 6, 3);\n",\
+"\n",\
+"freezone2\n",\
+"{ 1 P1 };\n",\
+"{ 1 P2 };\n",\
+"{ 1 P3 };\n",\
+"{ 1 P4 };\n",\
+"{ 1 P5 };\n",\
+"{ 1 P6 };\n",\
+"{ -0.1 P1, 0.3 P2, 0.3 P3, -0.1 P4, 0.3 P5, 0.3 P6 };\n",\
+"\n",\
+"freezonelimit\n",\
+"{ 1 P1 };\n",\
+"{ 1 P2 };\n",\
+"{ 1 P3 };\n",\
+"{ 1 P4 };\n",\
+"{ 1 P5 };\n",\
+"{ 1 P6 };\n",\
+"{ 0 P1, 0.25 P2, 0.25 P3, 0 P4, 0.25 P5, 0.25 P6 };\n",\
+"\n",\
+"freeset\n",\
+"1 2 3 5 6 7;\n",\
+"\n",\
+"freeset\n",\
+"1 3 4 6;\n",\
+"\n",\
+"\n",\
+"endrule\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"rule \"fill prism\"\n",\
+"\n",\
+"quality 1\n",\
+"\n",\
+"mappoints\n",\
+"(0, 0, 0);\n",\
+"(1, 0, 0);\n",\
+"(1, 1, 0);\n",\
+"(0, 1, 0);\n",\
+"(0.5, 0, -0.86);\n",\
+"(0.5, 1, -0.86);\n",\
+"\n",\
+"mapfaces\n",\
+"(1, 2, 3, 4) del;\n",\
+"(2, 5, 6, 3) del;\n",\
+"(5, 1, 4, 6) del;\n",\
+"(1, 5, 2) del;\n",\
+"(4, 3, 6) del;\n",\
+"\n",\
+"newpoints\n",\
+"\n",\
+"newfaces\n",\
+"\n",\
+"\n",\
+"elements\n",\
+"(1, 5, 2, 4, 6, 3);\n",\
+"\n",\
+"freezone2\n",\
+"{ 1 P1 };\n",\
+"{ 1 P2 };\n",\
+"{ 1 P3 };\n",\
+"{ 1 P4 };\n",\
+"{ 1 P5 };\n",\
+"{ 1 P6 };\n",\
+"\n",\
+"freeset\n",\
+"1 2 4 5;\n",\
+"\n",\
+"freeset\n",\
+"2 3 4 6;\n",\
+"\n",\
+"endrule\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"rule \"prism on 3 quad, one trig\"\n",\
+"\n",\
+"quality 2\n",\
+"\n",\
+"mappoints\n",\
+"(0, 0, 0);\n",\
+"(1, 0, 0);\n",\
+"(1, 1, 0);\n",\
+"(0, 1, 0);\n",\
+"(0.5, 0, -0.86);\n",\
+"(0.5, 1, -0.86);\n",\
+"\n",\
+"mapfaces\n",\
+"(1, 2, 3, 4) del;\n",\
+"(2, 5, 6, 3) del;\n",\
+"(5, 1, 4, 6) del;\n",\
+"(1, 5, 2) del;\n",\
+"\n",\
+"newpoints\n",\
+"\n",\
+"newfaces\n",\
+"(4, 6, 3);\n",\
+"\n",\
+"\n",\
+"elements\n",\
+"(1, 5, 2, 4, 6, 3);\n",\
+"\n",\
+"freezone2\n",\
+"{ 1 P1 };\n",\
+"{ 1 P2 };\n",\
+"{ 1 P3 };\n",\
+"{ 1 P4 };\n",\
+"{ 1 P5 };\n",\
+"{ 1 P6 };\n",\
+"\n",\
+"freeset\n",\
+"1 2 4 5;\n",\
+"\n",\
+"freeset\n",\
+"2 3 4 6;\n",\
+"\n",\
+"endrule\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"\n",\
+"rule \"flat prism\"\n",\
+"\n",\
+"quality 1\n",\
+"\n",\
+"mappoints\n",\
+"(0, 0, 0);\n",\
+"(1, 0, 0);\n",\
+"(0.5, 0.866, 0);\n",\
+"(0, 0, -1);\n",\
+"(1, 0, -1);\n",\
+"(0.5, 0.866, -1);\n",\
+"\n",\
+"\n",\
+"mapfaces\n",\
+"(1, 2, 3) del;\n",\
+"(5, 4, 6) del;\n",\
+"\n",\
+"newpoints\n",\
+"\n",\
+"newfaces\n",\
+"(1, 2, 4);\n",\
+"(4, 2, 5);\n",\
+"(2, 3, 5);\n",\
+"(5, 3, 6);\n",\
+"(3, 1, 6);\n",\
+"(6, 1, 4);\n",\
+"\n",\
+"\n",\
+"\n",\
+"elements\n",\
+"(1, 2, 3, 5, 4, 6);\n",\
+"\n",\
+"freezone2\n",\
+"{ 1 P1 };\n",\
+"{ 1 P2 };\n",\
+"{ 1 P5 };\n",\
+"{ 1 P6 };\n",\
+"endrule\n",\
+"\n",\
+0};
+}
diff --git a/Netgen/libsrc/meshing/quadrls.cpp b/Netgen/libsrc/meshing/quadrls.cpp
index 01555b72ea..57fb010144 100644
--- a/Netgen/libsrc/meshing/quadrls.cpp
+++ b/Netgen/libsrc/meshing/quadrls.cpp
@@ -118,9 +118,9 @@ const char * quadrules[] = {
 "\n",\
 "\n",\
 "\n",\
-"rule \"Quad P Right (1)\"\n",\
+"rule \"Quad P Right (2)\"\n",\
 "\n",\
-"quality 1\n",\
+"quality 2\n",\
 "\n",\
 "mappoints\n",\
 "(0, 0);\n",\
@@ -131,7 +131,7 @@ const char * quadrules[] = {
 "(1, 2) del;\n",\
 "\n",\
 "newpoints\n",\
-"(0, 1) { } { 1 y3 };\n",\
+"(0, 1) { -1 X2, 1 X3 } { 1 Y3 };\n",\
 "\n",\
 "newlines\n",\
 "(1, 4);\n",\
@@ -141,26 +141,31 @@ const char * quadrules[] = {
 "freearea\n",\
 "(0, 0);\n",\
 "(1, 0) { 1 X2 } { };\n",\
-"(1.2, 0.6) { 0.6 X2, 0.6 Y2 } { 0.6 Y3 };\n",\
+"(1.2, 0.5) { 0.7 X2, 0.5 X3 } { 0.5 Y3 };\n",\
 "(1, 1) { 1 X3 } { 1 Y3 };\n",\
-"(-0.5, 1.5) { } { 1.5 Y3 };\n",\
+"(-0.5, 1.5) { -2 X2, 1.5 X3 } { 1.5 Y3 };\n",\
 "\n",\
 "freearea2\n",\
 "(0, 0);\n",\
 "(1, 0) { 1 X2 } { };\n",\
-"(1, 0.5) { 0.5 X2, 0.5 Y2 } { 0.5 Y3 };\n",\
+"(1, 0.5) { 0.5 X2, 0.5 X3 } { 0.5 Y3 };\n",\
 "(1, 1) { 1 X3 } { 1 Y3 };\n",\
-"(0, 1) { } { 1 Y3 };\n",\
+"(0, 1) { -1 X2, 1 X3 } { 1 Y3 };\n",\
 "\n",\
 "elements\n",\
 "(1, 2, 3, 4);\n",\
 "\n",\
+"\n",\
+"orientations\n",\
+"(1, 2, 3);\n",\
+"\n",\
+"\n",\
 "endrule\n",\
 "\n",\
 "\n",\
-"rule \"Quad Right PL (1)\"\n",\
+"rule \"Quad Right PL (2)\"\n",\
 "\n",\
-"quality 1\n",\
+"quality 2\n",\
 "\n",\
 "mappoints\n",\
 "(0, 0);\n",\
@@ -182,9 +187,9 @@ const char * quadrules[] = {
 "(0, 0);\n",\
 "(1, 0) { 1 X2 } { };\n",\
 "(1, 1) { 1 X3 } { 1 Y3 };\n",\
-"(0.5, 1.2) { -0.2 X2, 0.6 X3, 0.6 X4 } { -0.2 Y2, 0.6 Y3, 0.6 Y4 };\n",\
+"(0.5, 1.2) { -0.1 X2, 0.6 X3, 0.6 X4 } { -0.1 Y2, 0.6 Y3, 0.6 Y4 };\n",\
 "(0, 1) { 1 X4 } { 1 Y4 };\n",\
-"(-0.2, 0.5) { -0.2 X2, -0.2 X3, 0.6 X4 } { -0.2 Y2, -0.2 Y3, 0.6 Y4 };\n",\
+"(-0.2, 0.5) { -0.1 X2, -0.1 X3, 0.6 X4 } { -0.1 Y2, -0.1 Y3, 0.6 Y4 };\n",\
 "\n",\
 "freearea2\n",\
 "(0, 0);\n",\
@@ -197,6 +202,13 @@ const char * quadrules[] = {
 "elements\n",\
 "(1, 2, 3, 4);\n",\
 "\n",\
+"orientations\n",\
+"(1, 2, 3);\n",\
+"(1, 3, 4);\n",\
+"(1, 2, 4);\n",\
+"(4, 2, 3);\n",\
+"\n",\
+"\n",\
 "endrule\n",\
 "\n",\
 "\n",\
@@ -246,9 +258,9 @@ const char * quadrules[] = {
 "\n",\
 "\n",\
 "\n",\
-"rule \"Left P Quad (1)\"\n",\
+"rule \"Left P Quad (2)\"\n",\
 "\n",\
-"quality 1\n",\
+"quality 2\n",\
 "\n",\
 "mappoints\n",\
 "(0, 0);\n",\
@@ -289,9 +301,9 @@ const char * quadrules[] = {
 "\n",\
 "\n",\
 "\n",\
-"rule \"Left Quad RP (1)\"\n",\
+"rule \"Left Quad RP (2)\"\n",\
 "\n",\
-"quality 1\n",\
+"quality 2\n",\
 "\n",\
 "mappoints\n",\
 "(0, 0);\n",\
@@ -312,9 +324,9 @@ const char * quadrules[] = {
 "freearea\n",\
 "(0, 0);\n",\
 "(1, 0) { 1 X2 } { };\n",\
-"(1.2, 0.5) { 0.6 X2, 0.6 X4, -0.2 X3 } { 0.6 Y2, 0.6 Y4, -0.2 Y3 };\n",\
+"(1.2, 0.5) { 0.6 X2, 0.6 X4, -0.1 X3 } { 0.6 Y2, 0.6 Y4, -0.1 Y3 };\n",\
 "(1, 1) { 1 X4 } { 1 Y4 };\n",\
-"(0.5, 1.2) { -0.2 X2, 0.6 X3, 0.6 X4 } { -0.2 Y2, 0.6 Y3, 0.6 Y4 };\n",\
+"(0.5, 1.2) { -0.1 X2, 0.6 X3, 0.6 X4 } { -0.1 Y2, 0.6 Y3, 0.6 Y4 };\n",\
 "(0, 1) { 1 X3 } { 1 Y3 };\n",\
 "\n",\
 "freearea2\n",\
@@ -328,6 +340,10 @@ const char * quadrules[] = {
 "elements\n",\
 "(1, 2, 4, 3);\n",\
 "\n",\
+"orientations\n",\
+"(1, 2, 4);\n",\
+"(1, 4, 3);\n",\
+"\n",\
 "endrule\n",\
 "\n",\
 "\n",\
@@ -651,9 +667,9 @@ const char * quadrules[] = {
 "\n",\
 "\n",\
 "\n",\
-"rule \"Vis A Vis (1)\"\n",\
+"rule \"Vis A Vis (2)\"\n",\
 "\n",\
-"quality 1\n",\
+"quality 2\n",\
 "\n",\
 "mappoints\n",\
 "(0, 0);\n",\
@@ -690,6 +706,12 @@ const char * quadrules[] = {
 "elements\n",\
 "(1, 2, 3, 4);\n",\
 "\n",\
+"orientations\n",\
+"(1, 3, 4);\n",\
+"(2, 3, 4);\n",\
+"(1, 2, 3);\n",\
+"(1, 2, 4);\n",\
+"\n",\
 "endrule\n",\
 "\n",\
 "\n",\
diff --git a/Netgen/libsrc/meshing/refine.cpp b/Netgen/libsrc/meshing/refine.cpp
index c14979693e..f7578d3394 100644
--- a/Netgen/libsrc/meshing/refine.cpp
+++ b/Netgen/libsrc/meshing/refine.cpp
@@ -4,96 +4,77 @@
 
 namespace netgen
 {
-  void Refinement :: Refine (Mesh & mesh, int /* levels */)
+  void Refinement :: Refine (Mesh & mesh)
   {
-    int i, j, k;
-
-
     // reduce 2nd order
     mesh.ComputeNVertices();
     mesh.SetNP(mesh.GetNV());
 
 
-
-    //  for (l = 1; l <= levels; l++)
-    //    {
     INDEX_2_HASHTABLE<int> between(mesh.GetNP() + 5);
       
     int oldne, oldns, oldnf;
 
     // refine edges
 
-
-    ARRAY<EdgePointGeomInfo> epgi;
+    ARRAY<EdgePointGeomInfo,PointIndex::BASE> epgi;
 
     oldns = mesh.GetNSeg();
-
-    for (i = 1; i <= oldns; i++)
+    for (SegmentIndex si = 0; si < oldns; si++)
       {
-	const Segment & el = mesh.LineSegment(i);
-
-	INDEX_2 i2(el.p1, el.p2);
-	i2.Sort();
+	const Segment & el = mesh.LineSegment(si);
 
-	int pnew;
+	INDEX_2 i2 = INDEX_2::Sort(el.p1, el.p2);
+	PointIndex pinew;
 	EdgePointGeomInfo ngi;
 
 	if (between.Used(i2))
 	  {
-	    pnew = between.Get(i2);
-	    ngi = epgi.Get(pnew);
+	    pinew = between.Get(i2);
+	    ngi = epgi[pinew]; 
 	  }
 	else
 	  {
-	    Point3d pb;
-	    pb = Center (mesh.Point (el.p1),
-			 mesh.Point (el.p2));
+	    Point3d pnew;
 
 	    PointBetween (mesh.Point (el.p1),
 			  mesh.Point (el.p2), 0.5,
 			  el.surfnr1, el.surfnr2,
 			  el.epgeominfo[0], el.epgeominfo[1],
-			  pb, ngi);
+			  pnew, ngi);
 
-	    pnew = mesh.AddPoint (pb);
+	    pinew = mesh.AddPoint (pnew);
+	    between.Set (i2, pinew);
 
-	    between.Set (i2, pnew);
 
-	    if (pnew > epgi.Size())
-	      epgi.SetSize (pnew);
-	    epgi.Elem(pnew) = ngi;
-
-	    (*testout) << "ref edge, oldgi = " 
-		       << el.epgeominfo[0] << " - " << el.epgeominfo[1] 
-		       << ", new gi = " << ngi << endl;
-	    
+	    if (pinew >= epgi.Size()+PointIndex::BASE)
+	      epgi.SetSize (pinew+1-PointIndex::BASE);
+	    epgi[pinew] = ngi;
 	  }
 
 	Segment ns1 = el;
 	Segment ns2 = el;
-	ns1.p2 = pnew;
+	ns1.p2 = pinew;
 	ns1.epgeominfo[1] = ngi;
-	ns2.p1 = pnew;
+	ns2.p1 = pinew;
 	ns2.epgeominfo[0] = ngi;
 
-	mesh.LineSegment(i) = ns1;
+	mesh.LineSegment(si) = ns1;
 	mesh.AddSegment (ns2);
       }
 
-
-
-    (*testout) << "refine surfaces" << endl;
     // refine surface elements
+    ARRAY<PointGeomInfo,PointIndex::BASE> surfgi (8*mesh.GetNP());
+    for (int i = PointIndex::BASE;
+	 i < surfgi.Size()+PointIndex::BASE; i++)
+      surfgi[i].trignum = -1;
 
-    ARRAY<PointGeomInfo> surfgi (8*mesh.GetNP());
-    for (i = 1; i <= surfgi.Size(); i++)
-      surfgi.Elem(i).trignum = -1;
 
     oldnf = mesh.GetNSE();
-    for (i = 1; i <= oldnf; i++)
+    for (SurfaceElementIndex sei = 0; sei < oldnf; sei++)
       {
 	int j, k;
-	const Element2d & el = mesh.SurfaceElement(i);
+	const Element2d & el = mesh.SurfaceElement(sei);
 
 	switch (el.GetType())
 	  {
@@ -102,27 +83,25 @@ namespace netgen
 	    {
 	      ArrayMem<int,6> pnums(6);
 	      ArrayMem<PointGeomInfo,6> pgis(6);
-	      
+
 	      static int betw[3][3] =
 		{ { 2, 3, 4 },
 		  { 1, 3, 5 },
 		  { 1, 2, 6 } };
-	      
+
 	      for (j = 1; j <= 3; j++)
 		{
 		  pnums.Elem(j) = el.PNum(j);
 		  pgis.Elem(j) = el.GeomInfoPi(j);
 		}
-	      
+
 	      for (j = 0; j < 3; j++)
 		{
-		  int pi1 = pnums.Elem(betw[j][0]);
-		  int pi2 = pnums.Elem(betw[j][1]);
-		  
+		  PointIndex pi1 = pnums.Elem(betw[j][0]);
+		  PointIndex pi2 = pnums.Elem(betw[j][1]);
+
 		  INDEX_2 i2 (pi1, pi2);
 		  i2.Sort();
-		  
-
 
 		  Point3d pb;
 		  PointGeomInfo pgi;
@@ -146,47 +125,15 @@ namespace netgen
 		  if (surfgi.Size() < pnums.Elem(4+j))
 		    surfgi.SetSize (pnums.Elem(4+j));
 		  surfgi.Elem(pnums.Elem(4+j)) = pgis.Elem(4+j);
-		      
-
-		  /*
-		  if (between.Used(i2))
-		    {
-		      pnums.Elem(4+j) = between.Get(i2);
-		      pgis.Elem(4+j) = surfgi.Get(pnums.Elem(4+j));
-		    }
-		  else
-		    {
-		      Point3d pb;
-		      PointBetween (mesh.Point (pi1),
-				    mesh.Point (pi2), 0.5,
-				    mesh.GetFaceDescriptor(el.GetIndex ()).SurfNr(),
-				    el.GeomInfoPi (betw[j][0]),
-				    el.GeomInfoPi (betw[j][1]),
-				    pb, pgis.Elem(4+j));
-		    
-		      pnums.Elem(4+j) = mesh.AddPoint (pb);
-		    
-		      between.Set (i2, pnums.Get(4+j));
-		    
-		      if (surfgi.Size() < pnums.Elem(4+j))
-			surfgi.SetSize (pnums.Elem(4+j));
-		      surfgi.Elem(pnums.Elem(4+j)) = pgis.Elem(4+j);
-
-		      (*testout) << "ref face, oldgi = " 
-				 << el.GeomInfoPi (betw[j][0]) << " - " 
-				 << el.GeomInfoPi (betw[j][1])
-				 << ", new gi = " << pgis.Elem(4+j) << endl;
-
-		    }
-		  */
 		}
 
-	      static int reftab[4][3] = 
+
+	      static int reftab[4][3] =
 		{ { 1, 6, 5 },
 		  { 2, 4, 6 },
 		  { 3, 5, 4 },
 		  { 6, 4, 5 } };
-	    
+
 	      int ind = el.GetIndex();
 	      for (j = 0; j < 4; j++)
 		{
@@ -197,9 +144,9 @@ namespace netgen
 		      nel.GeomInfoPi(k) = pgis.Get(reftab[j][k-1]);
 		    }
 		  nel.SetIndex(ind);
-		
+
 		  if (j == 0)
-		    mesh.SurfaceElement(i) = nel;
+		    mesh.SurfaceElement(sei) = nel;
 		  else
 		    mesh.AddSurfaceElement(nel);
 		}
@@ -207,31 +154,32 @@ namespace netgen
 	    }
 	  case QUAD:
 	  case QUAD6:
+	  case QUAD8:
 	    {
 	      ArrayMem<int,9> pnums(9);
 	      ArrayMem<PointGeomInfo,9> pgis(9);
-	    
+
 	      static int betw[5][3] =
 		{ { 1, 2, 5 },
 		  { 2, 3, 6 },
 		  { 3, 4, 7 },
 		  { 1, 4, 8 },
 		  { 1, 3, 9 } };
-	    
+
 	      for (j = 1; j <= 4; j++)
 		{
 		  pnums.Elem(j) = el.PNum(j);
 		  pgis.Elem(j) = el.GeomInfoPi(j);
 		}
-	    
+
 	      for (j = 0; j < 5; j++)
 		{
 		  int pi1 = pnums.Elem(betw[j][0]);
 		  int pi2 = pnums.Elem(betw[j][1]);
-		
+
 		  INDEX_2 i2 (pi1, pi2);
 		  i2.Sort();
-		
+
 		  if (between.Used(i2))
 		    {
 		      pnums.Elem(5+j) = between.Get(i2);
@@ -246,19 +194,19 @@ namespace netgen
 				    el.GeomInfoPi (betw[j][0]),
 				    el.GeomInfoPi (betw[j][1]),
 				    pb, pgis.Elem(5+j));
-		    
+
 		      pnums.Elem(5+j) = mesh.AddPoint (pb);
-		    
+
 		      between.Set (i2, pnums.Get(5+j));
-		    
+		      
 		      if (surfgi.Size() < pnums.Elem(5+j))
 			surfgi.SetSize (pnums.Elem(5+j));
 		      surfgi.Elem(pnums.Elem(5+j)) = pgis.Elem(5+j);
 		    }
 		}
 
-	      static int reftab[4][4] = 
-		{ 
+	      static int reftab[4][4] =
+		{
 		  { 1, 5, 9, 8 },
 		  { 5, 2, 6, 9 },
 		  { 8, 9, 7, 4 },
@@ -274,9 +222,9 @@ namespace netgen
 		      nel.GeomInfoPi(k) = pgis.Get(reftab[j][k-1]);
 		    }
 		  nel.SetIndex(ind);
-		
+
 		  if (j == 0)
-		    mesh.SurfaceElement(i) = nel;
+		    mesh.SurfaceElement(sei) = nel;
 		  else
 		    mesh.AddSurfaceElement(nel);
 		}
@@ -286,58 +234,62 @@ namespace netgen
 	    PrintSysError ("Refine: undefined surface element type ", int(el.GetType()));
 	  }
       }
-    
-    (*testout) << "refine volume" << endl;
 
     // refine volume elements
     oldne = mesh.GetNE();
-    for (i = 1; i <= oldne; i++)
+    for (ElementIndex ei = 0; ei < oldne; ei++)
       {
 	int j, k;
-	const Element & el = mesh.VolumeElement(i);
-	ArrayMem<int,10> pnums(10);
-	static int betw[6][3] =
-	  { { 1, 2, 5 },
-	    { 1, 3, 6 },
-	    { 1, 4, 7 },
-	    { 2, 3, 8 },
-	    { 2, 4, 9 },
-	    { 3, 4, 10 } };
-	  
-	int elrev = el.flags.reverse;
-
-	for (j = 1; j <= 4; j++)
-	  pnums.Elem(j) = el.PNum(j);
-	if (elrev)
-	  Swap (pnums.Elem(3), pnums.Elem(4));
 
-	for (j = 0; j < 6; j++)
+	const Element & el = mesh.VolumeElement(ei);
+	switch (el.GetType())
 	  {
-	    INDEX_2 i2;
-	    i2.I1() = pnums.Get(betw[j][0]);
-	    i2.I2() = pnums.Get(betw[j][1]);
-	    i2.Sort();
-
-	    if (between.Used(i2))
-	      pnums.Elem(5+j) = between.Get(i2);
-	    else
-	      {
-		pnums.Elem(5+j) = mesh.AddPoint
+	  case TET:
+	  case TET10:
+	    {
+	     ArrayMem<int,10> pnums(10);
+	     static int betw[6][3] =
+	     { { 1, 2, 5 },
+	       { 1, 3, 6 },
+	       { 1, 4, 7 },
+	       { 2, 3, 8 },
+	       { 2, 4, 9 },
+	       { 3, 4, 10 } };
+
+	     int elrev = el.flags.reverse;
+
+	     for (j = 1; j <= 4; j++)
+	     pnums.Elem(j) = el.PNum(j);
+	     if (elrev)
+	     swap (pnums.Elem(3), pnums.Elem(4));
+
+	     for (j = 0; j < 6; j++)
+	     {
+	       INDEX_2 i2;
+	       i2.I1() = pnums.Get(betw[j][0]);
+	       i2.I2() = pnums.Get(betw[j][1]);
+	       i2.Sort();
+
+	       if (between.Used(i2))
+	          pnums.Elem(5+j) = between.Get(i2);
+	       else
+	       {
+		  pnums.Elem(5+j) = mesh.AddPoint
 		  (Center (mesh.Point(i2.I1()),
 			   mesh.Point(i2.I2())));
-		between.Set (i2, pnums.Elem(5+j));
-	      }
-	  }
+		  between.Set (i2, pnums.Elem(5+j));
+	       }
+	    }
 
-	static int reftab[8][4] = 
-	  { { 1, 5, 6, 7 },
-	    { 5, 2, 8, 9 },
-	    { 6, 8, 3, 10 },
-	    { 7, 9, 10, 4 },
-	    { 5, 6, 7, 9 },
-	    { 5, 6, 9, 8 },
-	    { 6, 7, 9, 10 },
-	    { 6, 8, 10, 9 } };
+	    static int reftab[8][4] =
+	    { { 1, 5, 6, 7 },
+	      { 5, 2, 8, 9 },
+	      { 6, 8, 3, 10 },
+	      { 7, 9, 10, 4 },
+	      { 5, 6, 7, 9 },
+	      { 5, 6, 9, 8 },
+	      { 6, 7, 9, 10 },
+	      { 6, 8, 10, 9 } };
 	/*
 	  { { 1, 5, 6, 7 },
 	  { 5, 2, 8, 9 },
@@ -348,40 +300,264 @@ namespace netgen
 	  { 6, 7, 9, 10 },
 	  { 6, 8, 9, 10 } };
 	*/
-	static int reverse[8] =
-	  {
-	    0, 0, 0, 0, 0, 1, 0, 1
-	  };
-
-	int ind = el.GetIndex();
-	for (j = 0; j < 8; j++)
-	  {
-	    Element nel;
-	    for (k = 1; k <= 4; k++)
-	      nel.PNum(k) = pnums.Get(reftab[j][k-1]);
-	    nel.SetIndex(ind);
-	    nel.flags.reverse = reverse[j];
-	    if (elrev)
+	   static int reverse[8] =
+	   {
+	      0, 0, 0, 0, 0, 1, 0, 1
+	   };
+
+	   int ind = el.GetIndex();
+	   for (j = 0; j < 8; j++)
+	   {
+	      Element nel;
+	      for (k = 1; k <= 4; k++)
+	        nel.PNum(k) = pnums.Get(reftab[j][k-1]);
+	      nel.SetIndex(ind);
+	      nel.flags.reverse = reverse[j];
+	      if (elrev)
 	      {
 		nel.flags.reverse = 1 - nel.flags.reverse;
-		Swap (nel.PNum(3), nel.PNum(4));
+		swap (nel.PNum(3), nel.PNum(4));
 	      }
 
-	    if (j == 0)
-	      mesh.VolumeElement(i) = nel;
-	    else
-	      mesh.AddVolumeElement (nel);
+	      if (j == 0)
+	        mesh.VolumeElement(ei) = nel;
+	      else
+	        mesh.AddVolumeElement (nel);
+	    }
+	    break;
+          }
+          case HEX:
+          {
+	     ArrayMem<int,27> pnums(27);
+	     static int betw[13][3] =
+	     { { 1, 2, 9 },
+	       { 3, 4, 10 },
+	       { 4, 1, 11 },
+               { 2, 3, 12 },
+	       { 5, 6, 13 },
+	       { 7, 8, 14 },
+	       { 8, 5, 15 },
+	       { 6, 7, 16 },
+	       { 1, 5, 17 },
+	       { 2, 6, 18 },
+	       { 3, 7, 19 },
+	       { 4, 8, 20 },
+	       { 2, 8, 21 },
+	       };
+
+	     static int fbetw[12][3] =
+	     { { 1, 3, 22 },
+	       { 2, 4, 22 },
+	       { 5, 7, 23 },
+               { 6, 8, 23 },
+	       { 1, 6, 24 },
+	       { 2, 5, 24 },
+	       { 2, 7, 25 },
+	       { 3, 6, 25 },
+	       { 3, 8, 26 },
+	       { 4, 7, 26 },
+	       { 1, 8, 27 },
+	       { 4, 5, 27 },
+	       };
+
+
+	     pnums = -1;
+
+	     for (j = 1; j <= 8; j++)
+	     pnums.Elem(j) = el.PNum(j);
+
+
+	     for (j = 0; j < 13; j++)
+	     {
+	       INDEX_2 i2;
+	       i2.I1() = pnums.Get(betw[j][0]);
+	       i2.I2() = pnums.Get(betw[j][1]);
+	       i2.Sort();
+
+	       if (between.Used(i2))
+	          pnums.Elem(9+j) = between.Get(i2);
+	       else
+	       {
+		  pnums.Elem(9+j) = mesh.AddPoint
+		  (Center (mesh.Point(i2.I1()),
+			   mesh.Point(i2.I2())));
+		  between.Set (i2, pnums.Elem(9+j));
+	       }
+	    }
+
+	    for (j = 0; j < 6; j++)
+	    {
+	       INDEX_2 i2a, i2b;
+	       i2a.I1() = pnums.Get(fbetw[2*j][0]);
+	       i2a.I2() = pnums.Get(fbetw[2*j][1]);
+	       i2a.Sort();
+	       i2b.I1() = pnums.Get(fbetw[2*j+1][0]);
+	       i2b.I2() = pnums.Get(fbetw[2*j+1][1]);
+	       i2b.Sort();
+
+	       if (between.Used(i2a))
+		 pnums.Elem(22+j) = between.Get(i2a);
+	       else if (between.Used(i2b))
+		 pnums.Elem(22+j) = between.Get(i2b);
+	       else
+		 {
+		   pnums.Elem(22+j) = mesh.AddPoint
+		     (Center (mesh.Point(i2a.I1()),
+			      mesh.Point(i2a.I2())));
+
+		   between.Set (i2a, pnums.Elem(22+j));
+		 }
+	    }
+
+	    static int reftab[8][8] =
+	    { { 1, 9, 22, 11, 17, 24, 21, 27 },
+	      { 9, 2, 12, 22, 24, 18, 25, 21 },
+	      { 11, 22, 10, 4, 27, 21, 26, 20},
+	      { 22, 12, 3, 10, 21, 25, 19, 26},
+	      { 17, 24, 21, 27, 5, 13, 23, 15},
+	      { 24, 18, 25, 21, 13, 6, 16, 23},
+	      { 27, 21, 26, 20, 15, 23, 14, 8},
+	      { 21, 25, 19, 26, 23, 16, 7, 14} };
+
+
+	   int ind = el.GetIndex();
+	   for (j = 0; j < 8; j++)
+	   {
+	      Element nel(HEX);
+	      for (k = 1; k <= 8; k++)
+	        nel.PNum(k) = pnums.Get(reftab[j][k-1]);
+	      nel.SetIndex(ind);
+
+              if (j == 0)
+	        mesh.VolumeElement(ei) = nel;
+	      else
+	        mesh.AddVolumeElement (nel);
+           }
+           break;
 	  }
+	  case PRISM:
+          {
+	     ArrayMem<int,18> pnums(18);
+	     static int betw[9][3] =
+	     { { 3, 1, 7 },
+	       { 1, 2, 8 },
+	       { 3, 2, 9 },
+               { 6, 4, 10 },
+	       { 4, 5, 11 },
+	       { 6, 5, 12 },
+	       { 1, 4, 13 },
+	       { 3, 6, 14 },
+	       { 2, 5, 15 },
+	       };
+
+	     static int fbetw[6][3] =
+	     { { 1, 6, 16 },
+	       { 3, 4, 16 },
+	       { 1, 5, 17 },
+               { 2, 4, 17 },
+	       { 2, 6, 18 },
+	       { 3, 5, 18 },
+	       };
+
+	     //int elrev = el.flags.reverse;
+	     pnums = -1;
+
+	     for (j = 1; j <= 6; j++)
+	     pnums.Elem(j) = el.PNum(j);
+	    // if (elrev)
+	    // swap (pnums.Elem(3), pnums.Elem(4));
+
+	     for (j = 0; j < 9; j++)
+	     {
+	       INDEX_2 i2;
+	       i2.I1() = pnums.Get(betw[j][0]);
+	       i2.I2() = pnums.Get(betw[j][1]);
+	       i2.Sort();
+
+	       if (between.Used(i2))
+	          pnums.Elem(7+j) = between.Get(i2);
+	       else
+	       {
+		  pnums.Elem(7+j) = mesh.AddPoint
+		  (Center (mesh.Point(i2.I1()),
+			   mesh.Point(i2.I2())));
+		  between.Set (i2, pnums.Elem(7+j));
+	       }
+	    }
+
+	    for (j = 0; j < 3; j++)
+	    {
+	       INDEX_2 i2a, i2b;
+	       i2a.I1() = pnums.Get(fbetw[2*j][0]);
+	       i2a.I2() = pnums.Get(fbetw[2*j][1]);
+	       i2a.Sort();
+	       i2b.I1() = pnums.Get(fbetw[2*j+1][0]);
+	       i2b.I2() = pnums.Get(fbetw[2*j+1][1]);
+	       i2b.Sort();
+
+	       if (between.Used(i2a))
+		 pnums.Elem(16+j) = between.Get(i2a);
+	       else if (between.Used(i2b))
+		 pnums.Elem(16+j) = between.Get(i2b);
+	       else
+		 {
+		   pnums.Elem(16+j) = mesh.AddPoint
+		     (Center (mesh.Point(i2a.I1()),
+			      mesh.Point(i2a.I2())));
+
+		   between.Set (i2a, pnums.Elem(16+j));
+		 }
+	    }
+
+
+	    static int reftab[8][6] =
+	    { { 1, 8, 7, 13, 17, 16 },
+	      { 7, 8, 9, 16, 17, 18 },
+	      { 7, 9, 3, 16, 18, 14 },
+	      { 8, 2, 9, 17, 15, 18 },
+	      { 13, 17, 16, 4, 11, 10 },
+	      { 16, 17, 18, 10, 11, 12 },
+	      { 16, 18, 14, 10, 12, 6 },
+	      { 17, 15, 18, 11, 5, 12 } };
+
+
+	   int ind = el.GetIndex();
+	   for (j = 0; j < 8; j++)
+	   {
+	      Element nel(PRISM);
+	      for (k = 1; k <= 6; k++)
+	        nel.PNum(k) = pnums.Get(reftab[j][k-1]);
+	      nel.SetIndex(ind);
+
+
+	      //nel.flags.reverse = reverse[j];
+	      //if (elrev)
+	     // {
+		//nel.flags.reverse = 1 - nel.flags.reverse;
+		//swap (nel.PNum(3), nel.PNum(4));
+
+
+	      if (j == 0)
+	        mesh.VolumeElement(ei) = nel;
+	      else
+	        mesh.AddVolumeElement (nel);
+           }
+           break;
+	  }
+	  default:
+	    PrintSysError ("Refine: undefined volume element type ", int(el.GetType()));
+        }
       }
-      
+
+
     // update identification tables
-    for (i = 1; i <= mesh.GetIdentifications().GetMaxNr(); i++)
+    for (int i = 1; i <= mesh.GetIdentifications().GetMaxNr(); i++)
       {
 	ARRAY<int,PointIndex::BASE> identmap;
 	mesh.GetIdentifications().GetMap (i, identmap);
 
-	for (j = 1; j <= between.GetNBags(); j++)
-	  for (k = 1; k <= between.GetBagSize(j); k++)
+	for (int j = 1; j <= between.GetNBags(); j++)
+	  for (int k = 1; k <= between.GetBagSize(j); k++)
 	    {
 	      INDEX_2 i2;
 	      int newpi;
@@ -398,13 +574,12 @@ namespace netgen
 
       }
 
-
-    
-      
+    mesh.ComputeNVertices();
+    return;
 
     int cnttrials = 10;
     int wrongels = 0;
-    for (i = 1; i <= mesh.GetNE(); i++)
+    for (int i = 1; i <= mesh.GetNE(); i++)
       if (mesh.VolumeElement(i).Volume(mesh.Points()) < 0)
 	{
 	  wrongels++;
@@ -413,7 +588,6 @@ namespace netgen
       else
 	mesh.VolumeElement(i).flags.badel = 0;
 
-      
     if (wrongels)
       {
 	cout << "WARNING: " << wrongels << " with wrong orientation found" << endl;
@@ -421,12 +595,12 @@ namespace netgen
 	int np = mesh.GetNP();
 	ARRAY<Point3d> should(np);
 	ARRAY<Point3d> can(np);
-	for (i = 1; i <= np; i++)
+	for (int i = 1; i <= np; i++)
 	  {
 	    should.Elem(i) = can.Elem(i) = mesh.Point(i);
 	  }
-	for (i = 1; i <= between.GetNBags(); i++)
-	  for (j = 1; j <= between.GetBagSize(i); j++)
+	for (int i = 1; i <= between.GetNBags(); i++)
+	  for (int j = 1; j <= between.GetBagSize(i); j++)
 	    {
 	      INDEX_2 parent;
 	      int child;
@@ -437,10 +611,10 @@ namespace netgen
 
 	BitArray boundp(np);
 	boundp.Clear();
-	for (i = 1; i <= mesh.GetNSE(); i++)
+	for (int i = 1; i <= mesh.GetNSE(); i++)
 	  {
 	    const Element2d & sel = mesh.SurfaceElement(i);
-	    for (j = 1; j <= sel.GetNP(); j++)
+	    for (int j = 1; j <= sel.GetNP(); j++)
 	      boundp.Set(sel.PNum(j));
 	  }
 
@@ -457,10 +631,10 @@ namespace netgen
 
 		cout << "lam = " << lam << endl;
 
-		for (i = 1; i <= np; i++)
+		for (int i = 1; i <= np; i++)
 		  if (boundp.Test(i))
 		    {
-		      for (j = 1; j <= 3; j++)
+		      for (int j = 1; j <= 3; j++)
 			mesh.Point(i).X(j) = 
 			  lam * should.Get(i).X(j) +
 			  (1-lam) * can.Get(i).X(j);
@@ -471,38 +645,38 @@ namespace netgen
 
 		BitArray free (mesh.GetNP()), fhelp(mesh.GetNP());
 		free.Clear();
-		for (i = 1; i <= mesh.GetNE(); i++)
+		for (int i = 1; i <= mesh.GetNE(); i++)
 		  {
 		    const Element & el = mesh.VolumeElement(i);
 		    if (el.Volume(mesh.Points()) < 0)
-		      for (j = 1; j <= el.GetNP(); j++)
+		      for (int j = 1; j <= el.GetNP(); j++)
 			free.Set (el.PNum(j));
 		  }
-		for (k = 1; k <= 3; k++)
+		for (int k = 1; k <= 3; k++)
 		  {
 		    fhelp.Clear();
-		    for (i = 1; i <= mesh.GetNE(); i++)
+		    for (int i = 1; i <= mesh.GetNE(); i++)
 		      {
 			const Element & el = mesh.VolumeElement(i);
 			int freeel = 0;
-			for (j = 1; j <= el.GetNP(); j++)
+			for (int j = 1; j <= el.GetNP(); j++)
 			  if (free.Test(el.PNum(j)))
 			    freeel = 1;
 			if (freeel)
-			  for (j = 1; j <= el.GetNP(); j++)
+			  for (int j = 1; j <= el.GetNP(); j++)
 			    fhelp.Set (el.PNum(j));
 		      }
 		    free.Or (fhelp);
 		  }
 
 		(*testout) << "smooth points: " << endl;
-		for (i = 1; i <= free.Size(); i++)
+		for (int i = 1; i <= free.Size(); i++)
 		  if (free.Test(i))
 		    (*testout) << "p " << i << endl;
 
 		(*testout) << "surf points: " << endl;
-		for (i = 1; i <= mesh.GetNSE(); i++)
-		  for (j = 1; j <= 3; j++)
+		for (int i = 1; i <= mesh.GetNSE(); i++)
+		  for (int j = 1; j <= 3; j++)
 		    (*testout) << mesh.SurfaceElement(i).PNum(j) << endl;
 		  
 
@@ -514,14 +688,14 @@ namespace netgen
 
 
 		wrongels = 0;
-		for (i = 1; i <= mesh.GetNE(); i++)
+		for (int i = 1; i <= mesh.GetNE(); i++)
 		  {
 		    if (mesh.VolumeElement(i).Volume(mesh.Points()) < 0)
 		      {
 			wrongels++;
 			mesh.VolumeElement(i).flags.badel = 1;
 			(*testout) << "wrong el: ";
-			for (j = 1; j <= 4; j++)
+			for (int j = 1; j <= 4; j++)
 			  (*testout) << mesh.VolumeElement(i).PNum(j) << " ";
 			(*testout) << endl;
 		      }
@@ -532,7 +706,7 @@ namespace netgen
 	      }
 	    while (wrongels && cnttrials > 0);
 	  
-	    for (i = 1; i <= np; i++)
+	    for (int i = 1; i <= np; i++)
 	      can.Elem(i) = mesh.Point(i);
 	  }
       }
@@ -543,7 +717,5 @@ namespace netgen
       }
  
     mesh.ComputeNVertices();
-    (*testout) << "refine done" << endl;
-
   }
 }
diff --git a/Netgen/libsrc/meshing/ruler2.cpp b/Netgen/libsrc/meshing/ruler2.cpp
index 700a3c4f70..82c6dbe57d 100644
--- a/Netgen/libsrc/meshing/ruler2.cpp
+++ b/Netgen/libsrc/meshing/ruler2.cpp
@@ -57,8 +57,8 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
   double minelerr = 2 + 0.5 * tolerance * tolerance;
 
 
-  int ok, found;
-  netrule * rule;
+  bool ok;
+  int found;   // rule number
   Vector oldu, newu;
   Point2d np;
   Vec2d linevec;
@@ -172,18 +172,9 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
     }
 
 
-
-#ifdef MARK
-  MARK (applyrules2m1);
-#endif
-
   for (ri = 1; ri <= rules.Size(); ri++)
     {
-#ifdef MARK
-      MARK (applyrules2m1a);
-#endif
-
-      rule = rules.Get(ri);
+      netrule * rule = rules.Get(ri);
 
       if (loctestmode)
 	(*testout) << "Rule " << rule->Name() << endl;
@@ -192,7 +183,12 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
 
       pmap.SetSize (rule->GetNP());
       lmap.SetSize (rule->GetNL());
-
+      
+      lused = 0;
+      pused = 0;
+      pmap = 0;
+      lmap = 0;
+      /*
       for (i = 1; i <= lused.Size(); i++)
 	lused.Set (i, 0);
       for (i = 1; i <= pused.Size(); i++)
@@ -201,9 +197,10 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
 	pmap.Set(i, 0);
       for (i = 1; i <= lmap.Size(); i++)
 	lmap.Set(i, 0);
+      */
 
-      lused.Set (1, 1);
-      lmap.Set (1, 1);
+      lused[0] = 1;   // .Set (1, 1);
+      lmap[0] = 1;    // .Set (1, 1);
 
       for (j = 1; j <= 2; j++)
 	{
@@ -217,11 +214,6 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
       while (nlok >= 2)
 	{
 
-#ifdef MARK
-      MARK (applyrules2m2b);
-#endif
-
-
 	  if (nlok <= rule->GetNOldL())
 
 	    {
@@ -244,7 +236,7 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
 			lpoints.Get (loclin.I1()).Y();
 
 		      if (rule->CalcLineError (nlok, linevec) > maxerr)
-			ok = 0;
+			  ok = 0;
 
 		      for (j = 1; j <= 2 && ok; j++)
 			{
@@ -259,8 +251,7 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
 			    {
 			      if (rule->CalcPointDist (refpi, lpoints.Get(loclin.I(j))) > maxerr
 				  || !legalpoints.Get(loclin.I(j))
-				  || pused.Get(loclin.I(j)))
-				ok = 0;
+				  || pused.Get(loclin.I(j))) ok = 0;
 			    }
 			}
 		    }
@@ -296,11 +287,6 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
 
 	    {
 
-#ifdef MARK
-      MARK (applyrules2mc);
-#endif
-
-
 	      // all lines are mapped !!
 
 	      // map also all points:
@@ -335,7 +321,7 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
 			  if (pmap.Get(npok))
 			    pused.Elem(pmap.Get(npok))--;
 
-			  while (!ok && pmap.Get(npok) < maxlegalpoint /* lpoints.Size() */)
+			  while (!ok && pmap.Get(npok) < maxlegalpoint)
 			    {
 			      ok = 1;
 
@@ -349,8 +335,7 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
 				{
 				  if (rule->CalcPointDist (npok, lpoints.Get(pmap.Get(npok))) > maxerr 
 				      || !legalpoints.Get(pmap.Get(npok)) 
-				      )
-				    ok = 0;
+				      ) ok = 0;
 				}
 			    }
 
@@ -374,36 +359,61 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
 		  else
 
 		    {
-#ifdef MARK
-      MARK (applyrules2m2d);
-#endif
-
 		      if (ok)
-			{
-			  foundmap.Elem(ri)++;
-			}
+			foundmap.Elem(ri)++; 
 
 		      if (loctestmode)
 			(*testout) << "lines and points mapped" << endl;
 
-		      oldu.SetSize (2 * rule->GetNOldP());
 
-		      for (i = 1; i <= rule->GetNOldP(); i++)
+		      ok = 1;
+
+		      // check orientations
+
+		      for (i = 1; i <= rule->GetNOrientations() && ok; i++)
 			{
-			  //			  ui = lpoints.Get(pmap.Get(i)) - rule->GetPoint(i);
-			  Vec2d ui(rule->GetPoint(i), lpoints.Get(pmap.Get(i)));
-			  oldu.Set (2*i-1, ui.X());
-			  oldu.Set (2*i  , ui.Y());
+			  if (CW (lpoints.Get(pmap.Get(rule->GetOrientation(i).i1)),
+				  lpoints.Get(pmap.Get(rule->GetOrientation(i).i2)),
+				  lpoints.Get(pmap.Get(rule->GetOrientation(i).i3))) )
+			    {
+			      ok = 0;
+			      if (loctestmode)
+				(*testout) << "Orientation " << i << " not ok" << endl;
+			    }
 			}
 
-		      rule -> SetFreeZoneTransformation (oldu, tolerance);
-
-		      ok = 1;
+		      if (ok)
+			{
+			  oldu.SetSize (2 * rule->GetNOldP());
+			  
+			  for (i = 1; i <= rule->GetNOldP(); i++)
+			    {
+			      Vec2d ui(rule->GetPoint(i), lpoints.Get(pmap.Get(i)));
+			      oldu.Set (2*i-1, ui.X());
+			      oldu.Set (2*i  , ui.Y());
+			    }
+			  
+			  rule -> SetFreeZoneTransformation (oldu, tolerance);
+			}
+		      
 
-		      if (!rule->ConvexFreeZone())
+		      if (ok && !rule->ConvexFreeZone())
 			{
 			  ok = 0;
 			  if (loctestmode) (*testout) << "freezone not convex" << endl;
+
+			  /*
+			  static int cnt = 0;
+			  cnt++;
+			  if (cnt % 100 == 0)
+			    {
+			      cout << "freezone not convex, cnt = " << cnt << "; rule = " << rule->Name() << endl;
+			      (*testout) << "freezone not convex, cnt = " << cnt << "; rule = " << rule->Name() << endl;
+			      (*testout) << "tol = " << tolerance << endl;
+			      (*testout) << "maxerr = " << maxerr << "; minerr = " << minelerr << endl;
+			      (*testout) << "freezone = " << rule->GetTransFreeZone() << endl;
+			    }
+			  */
 			}
 
 		      // check freezone:
@@ -455,7 +465,7 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
 			}
 
 
-
+		      /*
 		      // check orientations
 
 		      for (i = 1; i <= rule->GetNOrientations() && ok; i++)
@@ -469,7 +479,7 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
 				(*testout) << "Orientation " << i << " not ok" << endl;
 			    }
 			}
-
+		      */
 
 
 		      if (ok)
@@ -542,7 +552,7 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
 			  if (elerr < minelerr)
 			    {
 
-			      if (testmode)
+			      if (loctestmode)
 				{
 				  (*testout) << "rule = " << rule->Name() << endl;
 				  (*testout) << "class = " << tolerance << endl;
@@ -613,25 +623,19 @@ int Meshing2 ::ApplyRules (ARRAY<Point2d> & lpoints,
     }
 
 
-#ifdef MARK
-  MARK (applyrules2m3);
-#endif
-
-
   if (found)
     {
-      for (i = 1; i <= tempnewpoints.Size(); i++)
-	lpoints.Append (tempnewpoints.Get(i));
-      for (i = 1; i <= tempnewlines.Size(); i++)
-	llines.Append (tempnewlines.Get(i));
-      for (i = 1; i <= tempdellines.Size(); i++)
-	dellines.Append (tempdellines.Get(i));
-      for (i = 1; i <= tempelements.Size(); i++)
-	elements.Append (tempelements.Get(i));
-
-      //    (*testout) << "minelerr = " << minelerr << endl;
+      for (i = 0; i < tempnewpoints.Size(); i++)
+	lpoints.Append (tempnewpoints[i]);
+      for (i = 0; i < tempnewlines.Size(); i++)
+	llines.Append (tempnewlines[i]);
+      for (i = 0; i < tempdellines.Size(); i++)
+	dellines.Append (tempdellines[i]);
+      for (i = 0; i < tempelements.Size(); i++)
+	elements.Append (tempelements[i]);
     }
 
+
   return found;
 }
 
diff --git a/Netgen/libsrc/meshing/ruler2.hpp b/Netgen/libsrc/meshing/ruler2.hpp
index 3627df521e..4ef855d4ec 100644
--- a/Netgen/libsrc/meshing/ruler2.hpp
+++ b/Netgen/libsrc/meshing/ruler2.hpp
@@ -42,6 +42,8 @@ private:
   ///
   DenseMatrix oldutonewu, oldutofreearea, oldutofreearealimit;
   ///
+  ARRAY<DenseMatrix*> oldutofreearea_i;
+  ///
   MatrixFixWidth<3> freesetinequ;
 
   ///
@@ -95,21 +97,36 @@ public:
   ///
   void GetFreeZone (ARRAY<Point2d> & afreearea);
   ///
-  float CalcPointDist (int pi, const Point2d & p) const;
+
+  double CalcPointDist (int pi, const Point2d & p) const
+  {
+    double dx = p.X() - points.Get(pi).X();
+    double dy = p.Y() - points.Get(pi).Y();
+    const threefloat * tf = &tolerances.Get(pi);
+    return tf->f1 * dx * dx + tf->f2 * dx * dy + tf->f3 * dy * dy;
+  }
+
   ///
   float CalcLineError (int li, const Vec2d & v) const;
 
   ///
   void SetFreeZoneTransformation (const Vector & u, int tolclass);
+
   ///
-  int IsInFreeZone (const Point2d & p) const
+  bool IsInFreeZone (const Point2d & p) const
   {
     if (p.X() < fzminx || p.X() > fzmaxx ||
 	p.Y() < fzminy || p.Y() > fzmaxy) return 0;
-    return IsInFreeZone2 (p);
+
+    for (int i = 0; i < transfreezone.Size(); i++)
+      {
+	if (freesetinequ(i, 0) * p.X() + 
+	    freesetinequ(i, 1) * p.Y() +
+	    freesetinequ(i, 2) > 0) return 0;
+      }
+    return 1;
   }
-  ///
-  int IsInFreeZone2 (const Point2d & p) const;
+
   ///
   int IsLineInFreeZone (const Point2d & p1, const Point2d & p2) const
   {
diff --git a/Netgen/libsrc/meshing/ruler3.cpp b/Netgen/libsrc/meshing/ruler3.cpp
index 0d6c2dab1f..48ba5bc465 100644
--- a/Netgen/libsrc/meshing/ruler3.cpp
+++ b/Netgen/libsrc/meshing/ruler3.cpp
@@ -65,7 +65,6 @@ int Meshing3 :: ApplyRules
  )
 
 {
-  //  const int rotsym = 3;
   int i, j, k, ri, nfok, npok, incnpok, refpi, locpi, locfi, locfr;
   int hi, minn, hpi;
   float hf, err, minerr, teterr, minteterr;
diff --git a/Netgen/libsrc/meshing/secondorder.cpp b/Netgen/libsrc/meshing/secondorder.cpp
index 120ff024f1..d260eb20d7 100644
--- a/Netgen/libsrc/meshing/secondorder.cpp
+++ b/Netgen/libsrc/meshing/secondorder.cpp
@@ -19,24 +19,30 @@ namespace netgen
   
   void Refinement :: MakeSecondOrder (Mesh & mesh)
   {
-    int i, j;
+    int nseg, nse, ne;
+
+    mesh.ComputeNVertices();
+    mesh.SetNP(mesh.GetNV());
   
     INDEX_2_HASHTABLE<int> between(mesh.GetNP() + 5);
 
-    int nseg, nse, ne;
 
+    bool thinlayers = 0;
+    for (ElementIndex ei = 0; ei < mesh.GetNE(); ei++)
+      if (mesh[ei].GetType() == PRISM ||
+	  mesh[ei].GetType() == PRISM12)
+	thinlayers = 1;
+    
 
     nseg = mesh.GetNSeg();
-    for (i = 1; i <= nseg; i++)
+    for (SegmentIndex si = 0; si < nseg; si++)
       {
-	Segment & el = mesh.LineSegment(i);
-      
-	INDEX_2 i2 (el.p1, el.p2);
-	i2.Sort();
-      
-	int pnew;
+	Segment & el = mesh.LineSegment(si);
+
+	INDEX_2 i2 = INDEX_2::Sort (el.p1, el.p2);
+
 	if (between.Used(i2))
-	  pnew = between.Get(i2);
+	  el.pmid = between.Get(i2);
 	else
 	  {
 	    Point3d pb;
@@ -47,31 +53,35 @@ namespace netgen
 			  el.epgeominfo[0], el.epgeominfo[1],
 			  pb, ngi);
 	  
-	    pnew = mesh.AddPoint (pb);
-	    between.Set (i2, pnew);
+	    el.pmid = mesh.AddPoint (pb);
+	    between.Set (i2, el.pmid);
 	  }
-	el.pmid = pnew;
       }
 
     // refine surface elements
     nse = mesh.GetNSE();
-    for (i = 1; i <= nse; i++)
+    for (SurfaceElementIndex sei = 0; sei < nse; sei++)
       {
 	int j;
-	const Element2d & el = mesh.SurfaceElement(i);
+	const Element2d & el = mesh.SurfaceElement(sei);
 
-	int onp = el.GetNP();
+	int onp;
       
 	Element2d newel;
 	newel.SetIndex (el.GetIndex());
 
 	static int betw_trig[3][3] =
-	  { { 2, 3, 4 },
-	    { 1, 3, 5 },
-	    { 1, 2, 6 } };
+	  { { 1, 2, 3 },
+	    { 0, 2, 4 },
+	    { 0, 1, 5 } };
 	static int betw_quad6[2][3] =
-	  { { 1, 2, 5 },
-	    { 4, 3, 6 } };
+	  { { 0, 1, 4 },
+	    { 3, 2, 5 } };
+	static int betw_quad8[4][3] =
+	  { { 0, 1, 4 },
+	    { 3, 2, 5 },
+	    { 0, 3, 6 },
+	    { 1, 2, 7 } };
 	int (*betw)[3];
       
 	switch (el.GetType())
@@ -81,34 +91,43 @@ namespace netgen
 	    {
 	      betw = betw_trig;
 	      newel.SetType (TRIG6);
+	      onp = 3;
 	      break;
 	    }
 	  case QUAD:
-	  case QUAD6:
+	  case QUAD6: 
 	  case QUAD8:
 	    {
-	      betw = betw_quad6;
-	      newel.SetType (QUAD6);
+	      if (thinlayers)
+		{
+		  betw = betw_quad6;
+		  newel.SetType (QUAD6);
+		}
+	      else
+		{
+		  betw = betw_quad8;
+		  newel.SetType (QUAD8);
+		}
+	      onp = 4;
 	      break;
 	    }
 	  default:
 	    PrintSysError ("Unhandled element in secondorder:", int(el.GetType()));
 	  }
 
-	for (j = 1; j <= onp; j++)
-	  newel.PNum(j) = el.PNum(j);
+	for (j = 0; j < onp; j++)
+	  newel[j] = el[j];
       
 	int nnp = newel.GetNP();
 	for (j = 0; j < nnp-onp; j++)
 	  {
-	    int pi1 = newel.PNum(betw[j][0]);
-	    int pi2 = newel.PNum(betw[j][1]);
+	    int pi1 = newel[betw[j][0]];
+	    int pi2 = newel[betw[j][1]];
 	  
-	    INDEX_2 i2 (pi1, pi2);
-	    i2.Sort();
+	    INDEX_2 i2 = INDEX_2::Sort (pi1, pi2);
 	  
 	    if (between.Used(i2))
-	      newel.PNum(onp+1+j) = between.Get(i2);
+	      newel[onp+j] = between.Get(i2);
 	    else
 	      {
 		Point3d pb;
@@ -116,32 +135,30 @@ namespace netgen
 		PointBetween (mesh.Point (pi1),
 			      mesh.Point (pi2), 0.5, 
 			      mesh.GetFaceDescriptor(el.GetIndex ()).SurfNr(),
-			      el.GeomInfoPi (betw[j][0]),
-			      el.GeomInfoPi (betw[j][1]),
+			      el.GeomInfoPi (betw[j][0]+1),
+			      el.GeomInfoPi (betw[j][1]+1),
 			      pb, newgi);
 
-		newel.PNum(onp+1+j) = mesh.AddPoint (pb);
-		between.Set (i2, newel.PNum(onp+1+j));
+		newel[onp+j] = mesh.AddPoint (pb);
+		between.Set (i2, newel[onp+j]);
 	      }
 	  }
       
-	mesh.SurfaceElement(i) = newel;
+	mesh.SurfaceElement(sei) = newel;
       }
 
  
+    //    int i, j;
+
 
 
     // refine volume elements
     ne = mesh.GetNE();
-    for (i = 1; i <= ne; i++)
+    for (int i = 1; i <= ne; i++)
       {
 	int j;
 	const Element & el = mesh.VolumeElement(i);
-
-	if (el.GetType() != TET && el.GetType() != PRISM)
-	  continue;
-
-	int onp = el.GetNP();
+	int onp;
 
 	Element newel;
 	newel.SetIndex (el.GetIndex());
@@ -171,6 +188,7 @@ namespace netgen
 	    {
 	      betw = betw_tet;
 	      newel.SetType (TET10);
+	      onp = 4;
 	      break;
 	    }
 	  case PRISM:
@@ -178,6 +196,7 @@ namespace netgen
 	    {
 	      betw = betw_prism;
 	      newel.SetType (PRISM12);
+	      onp = 6;
 	      break;
 	    }
 	  default:
@@ -185,11 +204,11 @@ namespace netgen
 	  }
 
 
-	for (j = 1; j <= onp; j++)
+	for (int j = 1; j <= onp; j++)
 	  newel.PNum(j) = el.PNum(j);
 	int nnp = newel.GetNP();
 
-	for (j = 0; j < nnp-onp; j++)
+	for (int j = 0; j < nnp-onp; j++)
 	  {
 	    INDEX_2 i2(newel.PNum(betw[j][0]),
 		       newel.PNum(betw[j][1]));
@@ -209,42 +228,58 @@ namespace netgen
 	mesh.VolumeElement (i) = newel;
       }
 
-    /*
+
     // makes problems after linear mesh refinement, since
     // 2nd order identifications are not removed
     // update identification tables
-    for (i = 1; i <= mesh.GetIdentifications().GetMaxNr(); i++)
-    {
-    ARRAY<int> identmap;
-    mesh.GetIdentifications().GetMap (i, identmap);
-
-    for (j = 1; j <= between.GetNBags(); j++)
-    for (k = 1; k <= between.GetBagSize(j); k++)
-    {
-    INDEX_2 i2;
-    int newpi;
-    between.GetData (j, k, i2, newpi);
-    INDEX_2 oi2(identmap.Get(i2.I1()),
-    identmap.Get(i2.I2()));
-    oi2.Sort();
-    if (between.Used (oi2))
-    {
-    int onewpi = between.Get(oi2);
-    mesh.GetIdentifications().Add (newpi, onewpi, i);
-    }
-    }
-
-    }
-    */
+    for (int i = 1; i <= mesh.GetIdentifications().GetMaxNr(); i++)
+      {
+	ARRAY<int,PointIndex::BASE> identmap;
+	mesh.GetIdentifications().GetMap (i, identmap);
+
+	for (INDEX_2_HASHTABLE<int>::Iterator it = between.Begin();
+	     it != between.End(); it++)
+	  {
+	      INDEX_2 i2;
+	      int newpi;
+	      between.GetData (it, i2, newpi);
+	      INDEX_2 oi2(identmap.Get(i2.I1()),
+			  identmap.Get(i2.I2()));
+	      oi2.Sort();
+	      if (between.Used (oi2))
+		{
+		  int onewpi = between.Get(oi2);
+		  mesh.GetIdentifications().Add (newpi, onewpi, i);
+		}
+	  }
+
+	/*
+	for (int j = 1; j <= between.GetNBags(); j++)
+	  for (int k = 1; k <= between.GetBagSize(j); k++)
+	    {
+	      INDEX_2 i2;
+	      int newpi;
+	      between.GetData (j, k, i2, newpi);
+	      INDEX_2 oi2(identmap.Get(i2.I1()),
+			  identmap.Get(i2.I2()));
+	      oi2.Sort();
+	      if (between.Used (oi2))
+		{
+		  int onewpi = between.Get(oi2);
+		  mesh.GetIdentifications().Add (newpi, onewpi, i);
+		}
+	    }
+	*/
+      }
+
 
     //  mesh.mglevels++;
     int oldsize = mesh.mlbetweennodes.Size();
     mesh.mlbetweennodes.SetSize(mesh.GetNP());
-    for (i = oldsize+1; i <= mesh.GetNP(); i++)
-      {
-	mesh.mlbetweennodes.Elem(i).I1() = 0;
-	mesh.mlbetweennodes.Elem(i).I2() = 0;
-      }
+    for (int i = oldsize; i < mesh.GetNP(); i++)
+      mesh.mlbetweennodes[i] = INDEX_2(0,0);
+
+    /*
     for (i = 1; i <= between.GetNBags(); i++)
       for (j = 1; j <= between.GetBagSize(i); j++)
 	{
@@ -253,6 +288,13 @@ namespace netgen
 	  between.GetData (i, j, oldp, newp);
 	  mesh.mlbetweennodes.Elem(newp) = oldp;
 	}
+    */
+
+    for (INDEX_2_HASHTABLE<int>::Iterator it = between.Begin();
+	 it != between.End(); it++)
+      {
+	mesh.mlbetweennodes[between.GetData (it)] = between.GetHash(it);
+      }
 
     mesh.ComputeNVertices();
   
@@ -265,13 +307,13 @@ namespace netgen
     PrintMessage (3, "Validate mesh");
     int np = mesh.GetNP();
     int ne = mesh.GetNE();
-    int i, j;
+    // int i, j;
     ARRAY<INDEX_2> parents(np);
   
-    for (i = 1; i <= np; i++)
+    for (int i = 1; i <= np; i++)
       parents.Elem(i) = INDEX_2(0,0);
 
-    for (i = 1; i <= ne; i++)
+    for (int i = 1; i <= ne; i++)
       {
 	const Element & el = mesh.VolumeElement(i);
 	if (el.GetType() == TET10)
@@ -283,7 +325,7 @@ namespace netgen
 		{ 2, 3, 8 },
 		{ 2, 4, 9 },
 		{ 3, 4, 10 } };
-	    for (j = 0; j < 6; j++)
+	    for (int j = 0; j < 6; j++)
 	      {
 		int f1 = el.PNum (betweentab[j][0]);
 		int f2 = el.PNum (betweentab[j][1]);
@@ -302,7 +344,7 @@ namespace netgen
   ValidateRefinedMesh (Mesh & mesh, 
 		       ARRAY<INDEX_2> & parents)
   {
-    int i, j, k;
+    // int i, j, k;
   
     // homotopy method
 
@@ -310,7 +352,7 @@ namespace netgen
 
     int cnttrials = 100;
     int wrongels = 0;
-    for (i = 1; i <= ne; i++)
+    for (int i = 1; i <= ne; i++)
       if (mesh.VolumeElement(i).CalcJacobianBadness (mesh.Points()) > 1e10)
 	{
 	  wrongels++;
@@ -334,12 +376,12 @@ namespace netgen
 	ARRAY<Point3d> should(np);
 	ARRAY<Point3d> can(np);
 
-	for (i = 1; i <= np; i++)
+	for (int i = 1; i <= np; i++)
 	  {
 	    should.Elem(i) = can.Elem(i) = mesh.Point(i);
 	  }
 
-	for (i = 1; i <= parents.Size(); i++)
+	for (int i = 1; i <= parents.Size(); i++)
 	  {
 	    if (parents.Get(i).I1())
 	      can.Elem(i) = Center (can.Elem(parents.Get(i).I1()),
@@ -348,16 +390,16 @@ namespace netgen
 
 	BitArray boundp(np);
 	boundp.Clear();
-	for (i = 1; i <= mesh.GetNSE(); i++)
+	for (int i = 1; i <= mesh.GetNSE(); i++)
 	  {
 	    const Element2d & sel = mesh.SurfaceElement(i);
-	    for (j = 1; j <= sel.GetNP(); j++)
+	    for (int j = 1; j <= sel.GetNP(); j++)
 	      boundp.Set(sel.PNum(j));
 	  }
 
 
 	(*testout) << "bpoints:" << endl;
-	for (i = 1; i <= np; i++)
+	for (int i = 1; i <= np; i++)
 	  if (boundp.Test(i))
 	    (*testout) << i << endl;
 
@@ -379,10 +421,10 @@ namespace netgen
 		factry = lam + (1-lam) * facok;
 		cout << "trying: " << factry << endl;
 
-		for (i = 1; i <= np; i++)
+		for (int i = 1; i <= np; i++)
 		  if (boundp.Test(i))
 		    {
-		      for (j = 1; j <= 3; j++)
+		      for (int j = 1; j <= 3; j++)
 			mesh.Point(i).X(j) = 
 			  lam * should.Get(i).X(j) +
 			  (1-lam) * can.Get(i).X(j);
@@ -392,7 +434,7 @@ namespace netgen
 	      
 		//	      (*testout) << "bad els: " << endl;
 		wrongels = 0;
-		for (i = 1; i <= ne; i++)
+		for (int i = 1; i <= ne; i++)
 		  {
 		    if (!illegalels.Test(i) && 
 			mesh.VolumeElement(i).
@@ -425,14 +467,14 @@ namespace netgen
 	    mesh.ImproveMeshJacobian (OPT_WORSTCASE);	      
 	  
 	    facok = factry;
-	    for (i = 1; i <= np; i++)
+	    for (int i = 1; i <= np; i++)
 	      can.Elem(i) = mesh.Point(i);
 	  }
       }
 
 
       
-    for (i = 1; i <= ne; i++)
+    for (int i = 1; i <= ne; i++)
       {
 	if (illegalels.Test(i))
 	  {
diff --git a/Netgen/libsrc/meshing/smoothing2.cpp b/Netgen/libsrc/meshing/smoothing2.cpp
index 32e44b256d..d26b20ad5d 100644
--- a/Netgen/libsrc/meshing/smoothing2.cpp
+++ b/Netgen/libsrc/meshing/smoothing2.cpp
@@ -6,460 +6,569 @@
 namespace netgen
 {
 
-static const MeshOptimize2d * meshthis;
+  static const MeshOptimize2d * meshthis;
 
 
+#ifdef OLD
 
+  void CalcTriangleBadness (double x2, double x3, double y3, double metricweight,
+				   double h, double & badness, double & g1x, double & g1y)
+  {
+    // badness = sqrt(3.0) /36 * circumference^2 / area - 1 
+    // p1 = (0, 0), p2 = (x2, 0), p3 = (x3, y3);
 
-inline void CalcTriangleBadness (double x2, double x3, double y3, double metricweight,
-			  double h, double & badness, double & g1x, double & g1y)
-{
-  // badness = sqrt(3.0) /36 * circumference^2 / area - 1 
-  // p1 = (0, 0), p2 = (x2, 0), p3 = (x3, y3);
-
-  Vec2d v23;
-  double l12, l13, l23, cir, area;
-  static const double c = sqrt(3.0) / 36;
-  double c1, c2, c3, c4;
+    Vec2d v23;
+    double l12, l13, l23, cir, area;
+    static const double c = sqrt(3.0) / 36;
+    double c1, c2, c3, c4;
 
-  v23.X() = x3 - x2;
-  v23.Y() = y3;
+    v23.X() = x3 - x2;
+    v23.Y() = y3;
 
-  l12 = x2;
-  l13 = sqrt (x3*x3 + y3*y3);
-  l23 = v23.Length();
+    l12 = x2;
+    l13 = sqrt (x3*x3 + y3*y3);
+    l23 = v23.Length();
 
-  cir = l12 + l13 + l23;
-  area = 0.5 * x2 * y3;
+    cir = l12 + l13 + l23;
+    area = 0.5 * x2 * y3;
 
-  if (area <= 1e-24 * cir * cir)
-    {
-      g1x = 0;
-      g1y = 0;
-      badness = 1e10;
-      return;
-    }
+    if (area <= 1e-24 * cir * cir)
+      {
+	g1x = 0;
+	g1y = 0;
+	badness = 1e10;
+	return;
+      }
 
-  badness = c * cir * cir / area - 1;
+    badness = c * cir * cir / area - 1;
 
-  c1 = 2 * c * cir / area;
-  c2 = 0.5 * c * cir * cir / (area * area);
+    c1 = 2 * c * cir / area;
+    c2 = 0.5 * c * cir * cir / (area * area);
 
-  g1x = c1 * ( - 1 - x3 / l13) - c2 * (-v23.Y());
-  g1y = c1 * (     - y3 / l13) - c2 * ( v23.X());
+    g1x = c1 * ( - 1 - x3 / l13) - c2 * (-v23.Y());
+    g1y = c1 * (     - y3 / l13) - c2 * ( v23.X());
   
-  //  metricweight = 0.1;
-  if (metricweight > 0)
-    {
-      // area = (x2 - x1) * (y3 - y1) - (x3 - x1) * (y2 - y1);
-      // add:  metricweight * (area / h^2 + h^2 / area - 2)
+    //  metricweight = 0.1;
+    if (metricweight > 0)
+      {
+	// area = (x2 - x1) * (y3 - y1) - (x3 - x1) * (y2 - y1);
+	// add:  metricweight * (area / h^2 + h^2 / area - 2)
       
-      const double area = x2 * y3;
-      const double dareax1 = -y3; 
-      const double dareay1 = x3 - x2; 
+	const double area = x2 * y3;
+	const double dareax1 = -y3; 
+	const double dareay1 = x3 - x2; 
 
-      const double areahh = area / (h * h);
-      const double fac = metricweight * (areahh - 1 / areahh) / area;
+	const double areahh = area / (h * h);
+	const double fac = metricweight * (areahh - 1 / areahh) / area;
 
-      badness += metricweight * (areahh + 1 / areahh - 2);
-      g1x += fac * dareax1;
-      g1y += fac * dareay1; 
+	badness += metricweight * (areahh + 1 / areahh - 2);
+	g1x += fac * dareax1;
+	g1y += fac * dareay1; 
       
-      /*
-      // add: metricweight * (l1^2/h^2 + l2^2/h^2 + l3^2/h2 + h^2/l1^2 + h^2/l2^2 + h^2/l3^2 - 6)
-      double h2 = h*h;
-      double l1 = x2*x2;
-      double l2 = x3*x3+y3*y3;
-      double l3 = (x2-x3)*(x2-x3)+y3*y3;
-      double dl1dx = 2*(-x2);
-      double dl1dy = 0;
-      double dl2dx = -2*x3;
-      double dl2dy = -2*y3;
-
-      badness += (l1/h2 + l2/h2 + l3/h2 +h2/l1 + h2/l2 + h2/l3-6) * metricweight;
-
-      g1x += metricweight * (dl1dx/h2-h2/(l1*l1)*dl1dx + dl2dx/h2-h2/(l2*l2)*dl2dx);
-      g1y += metricweight * (dl1dy/h2-h2/(l1*l1)*dl1dy + dl2dy/h2-h2/(l2*l2)*dl2dy);
-      */
-    }
-}
+	/*
+	// add: metricweight * (l1^2/h^2 + l2^2/h^2 + l3^2/h2 + h^2/l1^2 + h^2/l2^2 + h^2/l3^2 - 6)
+	double h2 = h*h;
+	double l1 = x2*x2;
+	double l2 = x3*x3+y3*y3;
+	double l3 = (x2-x3)*(x2-x3)+y3*y3;
+	double dl1dx = 2*(-x2);
+	double dl1dy = 0;
+	double dl2dx = -2*x3;
+	double dl2dy = -2*y3;
+
+	badness += (l1/h2 + l2/h2 + l3/h2 +h2/l1 + h2/l2 + h2/l3-6) * metricweight;
+
+	g1x += metricweight * (dl1dx/h2-h2/(l1*l1)*dl1dx + dl2dx/h2-h2/(l2*l2)*dl2dx);
+	g1y += metricweight * (dl1dy/h2-h2/(l1*l1)*dl1dy + dl2dy/h2-h2/(l2*l2)*dl2dy);
+	*/
+      }
+  }
+
+#endif
+
+
+  /*
+  static const double c_trig = sqrt(3.0) / 12;
+  static const double c_trig4 = sqrt(3.0) / 3;
+  */
+  static const double c_trig = 0.14433756;
+  static const double c_trig4 = 0.57735026;
+
+  inline double CalcTriangleBadness (double x2, double x3, double y3, 
+				     double metricweight, double h)
+  {
+    // badness = sqrt(3.0) / 12 * (\sum l_i^2) / area - 1 
+    // p1 = (0, 0), p2 = (x2, 0), p3 = (x3, y3);
+
+    double cir_2 = (x2*x2 + x3*x3 + y3*y3 - x2*x3);
+    double area = x2 * y3;
+    
+    if (area <= 1e-24 * cir_2)
+      return 1e10;
+    
+    double badness = c_trig4 * cir_2 / area - 1;
+    
+    if (metricweight > 0)
+      {
+	// add:  metricweight * (area / h^2 + h^2 / area - 2)
+
+	double areahh = area / (h * h);
+	badness += metricweight * (areahh + 1 / areahh - 2);
+      }
+    return badness;
+  }
+
+
+  
+  inline void CalcTriangleBadness (double x2, double x3, double y3, double metricweight,
+				   double h, double & badness, double & g1x, double & g1y)
+  {
+    // old: badness = sqrt(3.0) /36 * circumference^2 / area - 1 
+    // badness = sqrt(3.0) / 12 * (\sum l_i^2) / area - 1 
+    // p1 = (0, 0), p2 = (x2, 0), p3 = (x3, y3);
+
+
+    double cir_2 = 2* (x2*x2 + x3*x3 + y3*y3 - x2*x3);
+    double area = 0.5 * x2 * y3;
+
+    if (area <= 1e-24 * cir_2)
+      {
+	g1x = 0;
+	g1y = 0;
+	badness = 1e10;
+	return;
+      }
+
+    badness = c_trig * cir_2 / area - 1;
+
+    double c1 = -2 * c_trig / area;
+    double c2 = 0.5 * c_trig * cir_2 / (area * area);
+    g1x = c1 * (x2 + x3) + c2 * y3;
+    g1y = c1 * (y3)      + c2 * (x2-x3); 
+
+    if (metricweight > 0)
+      {
+	// area = (x2 - x1) * (y3 - y1) - (x3 - x1) * (y2 - y1);
+	// add:  metricweight * (area / h^2 + h^2 / area - 2)
+      
+	double area = x2 * y3;
+	double dareax1 = -y3; 
+	double dareay1 = x3 - x2; 
+
+	double areahh = area / (h * h);
+	double fac = metricweight * (areahh - 1 / areahh) / area;
+
+	badness += metricweight * (areahh + 1 / areahh - 2);
+	g1x += fac * dareax1;
+	g1y += fac * dareay1; 
+      }
+  }
 
 
+
+
+
+
+
+
+
+#ifdef OLD
   double CalcTriangleBadness (const Point3d & p1, 
 			      const Point3d & p2, 
 			      const Point3d & p3,
 			      double metricweight,
 			      double h)
   {
-  double badness;
-  double g1x, g1y;
+    double badness;
+    double g1x, g1y;
   
-  Vec3d e1 (p1, p2);
-  Vec3d e2 (p1, p3);
+    Vec3d e1 (p1, p2);
+    Vec3d e2 (p1, p3);
   
-  double e1l = e1.Length() + 1e-24;
-  e1 /= e1l;
-  double e1e2 = (e1 * e2);
-  e2.Add (-e1e2, e1);
-  double e2l = e2.Length();
+    double e1l = e1.Length() + 1e-24;
+    e1 /= e1l;
+    double e1e2 = (e1 * e2);
+    e2.Add (-e1e2, e1);
+    double e2l = e2.Length();
   
-  CalcTriangleBadness ( e1l, e1e2, e2l,
-			metricweight, h, badness, g1x, g1y);
-  return badness;
-}
+    CalcTriangleBadness ( e1l, e1e2, e2l,
+			  metricweight, h, badness, g1x, g1y);
+    return badness;
+  }
+#endif
 
 
-double CalcTriangleBadness (const Point3d & p1, 
-			    const Point3d & p2, 
-			    const Point3d & p3,
-			    const Vec3d & n,
-			    double metricweight,
-			    double h)
-{
-  double badness;
-  double g1x, g1y;
+
+
+  double CalcTriangleBadness (const Point3d & p1, 
+			      const Point3d & p2, 
+			      const Point3d & p3,
+			      double metricweight,
+			      double h)
+  {
+    // badness = sqrt(3.0) / 12 * (\sum l_i^2) / area - 1 
+    // p1 = (0, 0), p2 = (x2, 0), p3 = (x3, y3);
+
+    Vec3d e12(p1,p2);
+    Vec3d e13(p1,p3);
+    Vec3d e23(p2,p3);
   
-  Vec3d v1 (p1, p2);
-  Vec3d v2 (p1, p3);
+    double l12_2 = e12.Length2();
+    double l13_2 = e13.Length2();
+    double l23_2 = e23.Length2();
 
-  Vec3d e1 = v1;
-  Vec3d e2 = v2;
+    double cir_2 = l12_2 + l13_2 + l23_2;
+    Vec3d area_v = Cross (e12, e13);
+    double area = 0.5 * area_v.Length();
 
-  e1 -= (e1 * n) * n;
-  e1 /= (e1.Length() + 1e-12);
-  e2 = Cross (n, e1);
+    if (area <= 1e-24 * cir_2)
+      return 1e10;
 
-  CalcTriangleBadness ( (e1 * v1), (e1 * v2), (e2 * v2), 
-			metricweight, h, badness, g1x, g1y);
-  return badness;
-}
+    double badness = c_trig * cir_2 / area - 1;
 
+    if (metricweight > 0)
+      {
+	// area = (x2 - x1) * (y3 - y1) - (x3 - x1) * (y2 - y1);
+	// add:  metricweight * (area / h^2 + h^2 / area - 2)
 
+	const double areahh = area / (h * h);
+	badness += metricweight * (areahh + 1 / areahh - 2);
+      }
 
+    return badness;
+  }
 
-static Point3d sp1; 
-static PointGeomInfo gi1;
-static Vec3d n, t1, t2;
-static ARRAY<SurfaceElementIndex> locelements(0);
-static ARRAY<int> locrots(0);
-static ARRAY<double> lochs(0);
-// static int locerr2;
-static double locmetricweight = 0;
-static double loch;
-static int surfi, surfi2;
-static int uselocalh;
 
+  double CalcTriangleBadness (const Point3d & p1, 
+			      const Point3d & p2, 
+			      const Point3d & p3,
+			      const Vec3d & n,
+			      double metricweight,
+			      double h)
+  {
+    Vec3d v1 (p1, p2);
+    Vec3d v2 (p1, p3);
 
-class Opti2SurfaceMinFunction : public MinFunction
-{
-  const Mesh & mesh;
-public:
-  Opti2SurfaceMinFunction (const Mesh & amesh)
-    : mesh(amesh)
-    { } ;
-  virtual double FuncGrad (const Vector & x, Vector & g) const;
-  virtual double Func (const Vector & x) const;
-};
-  
-double Opti2SurfaceMinFunction :: 
-Func (const Vector & x) const
-{
-  Vector g(x.Size());
-  return FuncGrad (x, g);
-}
+    Vec3d e1 = v1;
+    Vec3d e2 = v2;
+
+    e1 -= (e1 * n) * n;
+    e1 /= (e1.Length() + 1e-24);
+    e2 = Cross (n, e1);
+
+    return CalcTriangleBadness ( (e1 * v1), (e1 * v2), (e2 * v2), 
+				 metricweight, h);
+  }
 
 
-double Opti2SurfaceMinFunction :: 
-FuncGrad (const Vector & x, Vector & grad) const
-{
-  int j, roti;
-  Vec3d n, vgrad;
-  Point3d pp1;
-  Vec2d g1;
-  double badness, hbadness;
 
-  vgrad = 0;
-  badness = 0;
 
 
-  meshthis -> GetNormalVector (surfi, sp1, gi1, n);
+  static MeshPoint sp1; 
+  static PointGeomInfo gi1;
+  static Vec3d n, t1, t2;
+  static ARRAY<SurfaceElementIndex> locelements(0);
+  static ARRAY<int> locrots(0);
+  static ARRAY<double> lochs(0);
+  // static int locerr2;
+  static double locmetricweight = 0;
+  static double loch;
+  static int surfi, surfi2;
+  static int uselocalh;
 
-  pp1 = sp1;
-  pp1.Add2 (x.Get(1), t1, x.Get(2), t2);
 
-  //  meshthis -> ProjectPoint (surfi, pp1);
-  // meshthis -> GetNormalVector (surfi, pp1, n);
+  class Opti2SurfaceMinFunction : public MinFunction
+  {
+    const Mesh & mesh;
+  public:
+    Opti2SurfaceMinFunction (const Mesh & amesh)
+      : mesh(amesh)
+    { } ;
+    virtual double FuncGrad (const Vector & x, Vector & g) const;
+    virtual double Func (const Vector & x) const;
+  };
+  
+  double Opti2SurfaceMinFunction :: 
+  Func (const Vector & x) const
+  {
+    Vector g(x.Size());
+    return FuncGrad (x, g);
+  }
 
 
-  for (j = 0; j < locelements.Size(); j++)
-    {
-      roti = locrots[j];
-      const Element2d & bel = mesh[locelements[j]];
+  double Opti2SurfaceMinFunction :: 
+  FuncGrad (const Vector & x, Vector & grad) const
+  {
+    Vec3d n, vgrad;
+    Point3d pp1;
+    double g1x, g1y;
+    double badness, hbadness;
 
-      Vec3d e1(pp1, mesh[bel.PNumMod(roti + 1)]);
-      Vec3d e2(pp1, mesh[bel.PNumMod(roti + 2)]);
+    vgrad = 0;
+    badness = 0;
 
-      if (uselocalh) loch = lochs[j];
+    meshthis -> GetNormalVector (surfi, sp1, gi1, n);
 
-      double e1l = e1.Length();
-      if (Determinant(e1, e2, n) > 1e-8 * e1l * e2.Length())
-	{
-	  e1 /= e1l;
-	  double e1e2 = e1 * e2;
-	  e2.Add (-e1e2, e1);
-	  double e2l = e2.Length();
+    pp1 = sp1;
+    pp1.Add2 (x.Get(1), t1, x.Get(2), t2);
 
-	  CalcTriangleBadness ( e1l, e1e2, e2l, locmetricweight, loch,
-				hbadness, g1.X(), g1.Y());
+    //  meshthis -> ProjectPoint (surfi, pp1);
+    // meshthis -> GetNormalVector (surfi, pp1, n);
 
-	  badness += hbadness;
-	  vgrad.Add2 (g1.X(), e1, g1.Y() / e2l, e2);
-	}
-      else
-	badness += 1e8;
-    }
+    for (int j = 0; j < locelements.Size(); j++)
+      {
+	int roti = locrots[j];
+	const Element2d & bel = mesh[locelements[j]];
 
-  vgrad.Add (-(vgrad * n), n);
+	Vec3d e1(pp1, mesh[bel.PNumMod(roti + 1)]);
+	Vec3d e2(pp1, mesh[bel.PNumMod(roti + 2)]);
 
-  grad.Elem(1) = vgrad * t1;
-  grad.Elem(2) = vgrad * t2;
-  return badness;
-}
+	if (uselocalh) loch = lochs[j];
 
+	double e1l = e1.Length();
+	if (Determinant(e1, e2, n) > 1e-8 * e1l * e2.Length())
+	  {
+	    e1 /= e1l;
+	    double e1e2 = e1 * e2;
+	    e2.Add (-e1e2, e1);
+	    double e2l = e2.Length();
 
-class Opti2EdgeMinFunction : public MinFunction
-{
-  const Mesh & mesh;
-public:
-  Opti2EdgeMinFunction (const Mesh & amesh)
-    : mesh(amesh) { } ;
+	    CalcTriangleBadness ( e1l, e1e2, e2l, locmetricweight, loch,
+				  hbadness, g1x, g1y);
 
-  virtual double FuncGrad (const Vector & x, Vector & g) const;
-  virtual double Func (const Vector & x) const;
-};
+	    badness += hbadness;
+	    vgrad.Add2 (g1x, e1, g1y / e2l, e2);
+	  }
+	else
+	  badness += 1e8;
+      }
 
-double Opti2EdgeMinFunction :: Func (const Vector & x) const
-{
-  Vector g(x.Size());
-  return FuncGrad (x, g);
-}
+    vgrad.Add (-(vgrad * n), n);
 
-double Opti2EdgeMinFunction :: FuncGrad (const Vector & x, Vector & grad) const
-{
-  int j, rot;
-  Vec3d n1, n2, v1, v2, e1, e2, vgrad;
-  Point3d pp1;
-  Vec2d g1;
-  double badness, hbadness;
+    grad.Elem(1) = vgrad * t1;
+    grad.Elem(2) = vgrad * t2;
+    return badness;
+  }
 
-  vgrad.X() = 0;
-  vgrad.Y() = 0;
-  vgrad.Z() = 0;
-  badness = 0;
 
-  pp1 = sp1 + x.Get(1) * t1;
-  meshthis -> ProjectPoint2 (surfi, surfi2, pp1);
+  class Opti2EdgeMinFunction : public MinFunction
+  {
+    const Mesh & mesh;
+  public:
+    Opti2EdgeMinFunction (const Mesh & amesh)
+      : mesh(amesh) { } ;
 
-  for (j = 0; j < locelements.Size(); j++)
-    {
-      rot = locrots[j];
-      const Element2d & bel = mesh[locelements[j]];
+    virtual double FuncGrad (const Vector & x, Vector & g) const;
+    virtual double Func (const Vector & x) const;
+  };
 
-      v1 = mesh[bel.PNumMod(rot + 1)] - pp1;
-      v2 = mesh[bel.PNumMod(rot + 2)] - pp1;
+  double Opti2EdgeMinFunction :: Func (const Vector & x) const
+  {
+    Vector g(x.Size());
+    return FuncGrad (x, g);
+  }
 
-      e1 = v1;
-      e2 = v2;
-      e1 /= e1.Length();
-      e2 -= (e1 * e2) * e1;
-      e2 /= e2.Length();
+  double Opti2EdgeMinFunction :: FuncGrad (const Vector & x, Vector & grad) const
+  {
+    int j, rot;
+    Vec3d n1, n2, v1, v2, e1, e2, vgrad;
+    Point3d pp1;
+    Vec2d g1;
+    double badness, hbadness;
 
-      if (uselocalh) loch = lochs[j];
-      CalcTriangleBadness ( (e1 * v1), (e1 * v2), (e2 * v2), locmetricweight, loch,
-			    hbadness, g1.X(), g1.Y());
+    vgrad.X() = 0;
+    vgrad.Y() = 0;
+    vgrad.Z() = 0;
+    badness = 0;
 
-      badness += hbadness;
+    pp1 = sp1 + x.Get(1) * t1;
+    meshthis -> ProjectPoint2 (surfi, surfi2, pp1);
 
-      vgrad.X() += g1.X() * e1.X() + g1.Y() * e2.X();
-      vgrad.Y() += g1.X() * e1.Y() + g1.Y() * e2.Y();
-      vgrad.Z() += g1.X() * e1.Z() + g1.Y() * e2.Z();
-    }
+    for (j = 0; j < locelements.Size(); j++)
+      {
+	rot = locrots[j];
+	const Element2d & bel = mesh[locelements[j]];
 
-  meshthis -> GetNormalVector (surfi, pp1, n1);
-  meshthis -> GetNormalVector (surfi2, pp1, n2);
+	v1 = mesh[bel.PNumMod(rot + 1)] - pp1;
+	v2 = mesh[bel.PNumMod(rot + 2)] - pp1;
 
-  v1 = Cross (n1, n2);
-  v1 /= v1.Length();
+	e1 = v1;
+	e2 = v2;
+	e1 /= e1.Length();
+	e2 -= (e1 * e2) * e1;
+	e2 /= e2.Length();
 
-  grad.Elem(1) = (vgrad * v1) * (t1 * v1);
+	if (uselocalh) loch = lochs[j];
+	CalcTriangleBadness ( (e1 * v1), (e1 * v2), (e2 * v2), locmetricweight, loch,
+			      hbadness, g1.X(), g1.Y());
 
-  return badness;
-}
+	badness += hbadness;
 
+	vgrad.X() += g1.X() * e1.X() + g1.Y() * e2.X();
+	vgrad.Y() += g1.X() * e1.Y() + g1.Y() * e2.Y();
+	vgrad.Z() += g1.X() * e1.Z() + g1.Y() * e2.Z();
+      }
 
+    meshthis -> GetNormalVector (surfi, pp1, n1);
+    meshthis -> GetNormalVector (surfi2, pp1, n2);
 
+    v1 = Cross (n1, n2);
+    v1 /= v1.Length();
 
-class Opti2SurfaceMinFunctionJacobian : public MinFunction
-{
-  const Mesh & mesh;
-public:
-  Opti2SurfaceMinFunctionJacobian (const Mesh & amesh)
-    : mesh(amesh)
+    grad.Elem(1) = (vgrad * v1) * (t1 * v1);
+
+    return badness;
+  }
+
+
+
+
+  class Opti2SurfaceMinFunctionJacobian : public MinFunction
+  {
+    const Mesh & mesh;
+  public:
+    Opti2SurfaceMinFunctionJacobian (const Mesh & amesh)
+      : mesh(amesh)
     { } ;
-  virtual double FuncGrad (const Vector & x, Vector & g) const;
-  virtual double Func (const Vector & x) const;
-};
+    virtual double FuncGrad (const Vector & x, Vector & g) const;
+    virtual double FuncDeriv (const Vector & x, const Vector & dir, double & deriv) const;
+    virtual double Func (const Vector & x) const;
+  };
   
-double Opti2SurfaceMinFunctionJacobian :: 
-Func (const Vector & x) const
-{
-  Vector g(x.Size());
-  return FuncGrad (x, g);
-}
+  double Opti2SurfaceMinFunctionJacobian :: 
+  Func (const Vector & x) const
+  {
+    Vector g(x.Size());
+    return FuncGrad (x, g);
+  }
 
 
-double Opti2SurfaceMinFunctionJacobian :: 
-FuncGrad (const Vector & x, Vector & grad) const
-{
-  // from 2d:
+  double Opti2SurfaceMinFunctionJacobian :: 
+  FuncGrad (const Vector & x, Vector & grad) const
+  {
+    // from 2d:
 
-  int j, k, lpi, gpi;
-  Vec3d n, vgrad;
-  Point3d pp1;
-  Vec2d g1, vdir;
-  double badness, hbadness, hbad, hderiv;
+    int j, k, lpi, gpi;
+    Vec3d n, vgrad;
+    Point3d pp1;
+    Vec2d g1, vdir;
+    double badness, hbadness, hbad, hderiv;
 
-  vgrad = 0;
-  badness = 0;
+    vgrad = 0;
+    badness = 0;
 
-  meshthis -> GetNormalVector (surfi, sp1, gi1, n);
+    meshthis -> GetNormalVector (surfi, sp1, gi1, n);
 
-  pp1 = sp1;
-  pp1.Add2 (x.Get(1), t1, x.Get(2), t2);
+    pp1 = sp1;
+    pp1.Add2 (x.Get(1), t1, x.Get(2), t2);
 
-  //  meshthis -> ProjectPoint (surfi, pp1);
-  //  meshthis -> GetNormalVector (surfi, pp1, n);
+    //  meshthis -> ProjectPoint (surfi, pp1);
+    //  meshthis -> GetNormalVector (surfi, pp1, n);
 
-  static ARRAY<Point2d> pts2d;
-  pts2d.SetSize(mesh.GetNP());
+    static ARRAY<Point2d> pts2d;
+    pts2d.SetSize(mesh.GetNP());
 
-  grad = 0;
+    grad = 0;
 
-  for (j = 1; j <= locelements.Size(); j++)
-    {
-      lpi = locrots.Get(j);
-      const Element2d & bel = 
-	mesh[locelements.Get(j)];
+    for (j = 1; j <= locelements.Size(); j++)
+      {
+	lpi = locrots.Get(j);
+	const Element2d & bel = 
+	  mesh[locelements.Get(j)];
       
-      gpi = bel.PNum(lpi);
+	gpi = bel.PNum(lpi);
 
-      for (k = 1; k <= bel.GetNP(); k++)
-	{
-	  PointIndex pi = bel.PNum(k);
-	  pts2d.Elem(pi) = Point2d (t1 * (mesh.Point(pi) - sp1), 
-				    t2 * (mesh.Point(pi) - sp1)); 
-	}				    
-      pts2d.Elem(gpi) = Point2d (x.Get(1), x.Get(2));
+	for (k = 1; k <= bel.GetNP(); k++)
+	  {
+	    PointIndex pi = bel.PNum(k);
+	    pts2d.Elem(pi) = Point2d (t1 * (mesh.Point(pi) - sp1), 
+				      t2 * (mesh.Point(pi) - sp1)); 
+	  }				    
+	pts2d.Elem(gpi) = Point2d (x.Get(1), x.Get(2));
       
 
-      for (k = 1; k <= 2; k++)
-	{
-	  if (k == 1)
-	    vdir = Vec2d (1, 0);
-	  else
-	    vdir = Vec2d (0, 1);
+	for (k = 1; k <= 2; k++)
+	  {
+	    if (k == 1)
+	      vdir = Vec2d (1, 0);
+	    else
+	      vdir = Vec2d (0, 1);
 	  
-	  hbad = bel.
-	    CalcJacobianBadnessDirDeriv (pts2d, lpi, vdir, hderiv);
+	    hbad = bel.
+	      CalcJacobianBadnessDirDeriv (pts2d, lpi, vdir, hderiv);
 
-	  grad.Elem(k) += hderiv;
-	  if (k == 1)
-	    badness += hbad;
-	}
-	  
-	  
-      /*
-      Vec3d e1(pp1, mesh.Point(bel.PNumMod(lpi + 1)));
-      Vec3d e2(pp1, mesh.Point(bel.PNumMod(lpi + 2)));
+	    grad.Elem(k) += hderiv;
+	    if (k == 1)
+	      badness += hbad;
+	  }
+      }
 
-      if (uselocalh)
-	loch = lochs.Get(j);
 
-      double e1l = e1.Length();
-      if (Determinant(e1, e2, n) > 1e-8 * e1l * e2.Length())
-	{
-	  e1 /= e1l;
-	  double e1e2 = e1 * e2;
-	  e2.Add (-e1e2, e1);
-	  double e2l = e2.Length();
+    /*
+      vgrad.Add (-(vgrad * n), n);
 
-	  CalcTriangleBadness ( e1l, e1e2, e2l, locmetricweight, loch,
-				hbadness, g1.X(), g1.Y());
+      grad.Elem(1) = vgrad * t1;
+      grad.Elem(2) = vgrad * t2;
+    */
+    return badness;
+  }
 
-	  badness += hbadness;
-	  vgrad.Add2 (g1.X(), e1, g1.Y() / e2l, e2);
-	}
-      else
-	badness += 1e8;
-      */
-    }
 
 
-  /*
-  vgrad.Add (-(vgrad * n), n);
 
-  grad.Elem(1) = vgrad * t1;
-  grad.Elem(2) = vgrad * t2;
-  */
-  return badness;
+  double Opti2SurfaceMinFunctionJacobian :: 
+  FuncDeriv (const Vector & x, const Vector & dir, double & deriv) const
+  {
+    // from 2d:
 
-  /*
+    int j, k, lpi, gpi;
+    Vec3d n, vgrad;
+    Point3d pp1;
+    Vec2d g1, vdir;
+    double badness, hbadness, hbad, hderiv;
 
-  // from 3d:
+    vgrad = 0;
+    badness = 0;
 
-  int j, k;
-  int lpi;
-  double badness = 0, hbad;
+    meshthis -> GetNormalVector (surfi, sp1, gi1, n);
 
-  Point3d hp = points.Elem(actpind);
-  points.Elem(actpind) = hp + Vec3d (x.Get(1), x.Get(2), x.Get(3));
+    pp1 = sp1;
+    pp1.Add2 (x.Get(1), t1, x.Get(2), t2);
 
-  double hderiv;
-  Vec3d vdir;
-  g.SetSize(3);
-  g = 0;
+    static ARRAY<Point2d> pts2d;
+    pts2d.SetSize(mesh.GetNP());
 
-  for (j = 1; j <= elementsonpoint.EntrySize(actpind); j++)
-    {
-      int eli = elementsonpoint.Get(actpind, j);
-      const Element & el = elements.Get(eli);
+    deriv = 0;
 
-      lpi = 0;
-      for (k = 1; k <= el.GetNP(); k++)
-	if (el.PNum(k) == actpind)
-	  lpi = k;
-      if (!lpi) cerr << "loc point not found" << endl;
+    for (j = 1; j <= locelements.Size(); j++)
+      {
+	lpi = locrots.Get(j);
+	const Element2d & bel = 
+	  mesh[locelements.Get(j)];
+      
+	gpi = bel.PNum(lpi);
 
-      for (k = 1; k <= 3; k++)
-	{
-	  vdir = Vec3d(0,0,0);
-	  vdir.X(k) = 1;
-
-	  hbad = elements.Get(eli).
-	    CalcJacobianBadnessDirDeriv (points, lpi, vdir, hderiv);
-	  g.Elem(k) += hderiv;
-	  if (k == 1)
-	    badness += hbad;
-	}
-    }
-  
-  points.Elem(actpind) = hp; 
+	for (k = 1; k <= bel.GetNP(); k++)
+	  {
+	    PointIndex pi = bel.PNum(k);
+	    pts2d.Elem(pi) = Point2d (t1 * (mesh.Point(pi) - sp1), 
+				      t2 * (mesh.Point(pi) - sp1)); 
+	  }				    
+	pts2d.Elem(gpi) = Point2d (x.Get(1), x.Get(2));
+      
 
-  return badness;
+	vdir = Vec2d (dir(0), dir(1));
+	  
+	hbad = bel.
+	  CalcJacobianBadnessDirDeriv (pts2d, lpi, vdir, hderiv);
+      
+	deriv += hderiv;
+	badness += hbad;
+      }
 
-  */
 
-}
+    return badness;
+  }
 
 
 
@@ -467,324 +576,347 @@ FuncGrad (const Vector & x, Vector & grad) const
 
 
 
-MeshOptimize2d dummy;
+  MeshOptimize2d dummy;
 
-MeshOptimize2d :: MeshOptimize2d ()
-{
-  SetFaceIndex (0);
-  SetImproveEdges (0);
-  SetMetricWeight (0);
-  SetWriteStatus (1);
-}
+  MeshOptimize2d :: MeshOptimize2d ()
+  {
+    SetFaceIndex (0);
+    SetImproveEdges (0);
+    SetMetricWeight (0);
+    SetWriteStatus (1);
+  }
 
 
-void MeshOptimize2d :: SelectSurfaceOfPoint (const Point3d & p,
-					     const PointGeomInfo & gi)
-{
-  ;
-}
+  void MeshOptimize2d :: SelectSurfaceOfPoint (const Point3d & p,
+					       const PointGeomInfo & gi)
+  {
+    ;
+  }
 
-void MeshOptimize2d :: ImproveMesh (Mesh & mesh)
-{
-  if (!faceindex)
-    {
-      PrintMessage (3, "Smoothing");
+  void MeshOptimize2d :: ImproveMesh (Mesh & mesh)
+  {
+    if (!faceindex)
+      {
+	PrintMessage (3, "Smoothing");
 
-      for (faceindex = 1; faceindex <= mesh.GetNFD(); faceindex++)
-	{
-	  ImproveMesh (mesh);
-	  if (multithread.terminate)
-	    throw NgException ("Meshing stopped");
-	}
-      faceindex = 0;
-      return;
-    }
+	for (faceindex = 1; faceindex <= mesh.GetNFD(); faceindex++)
+	  {
+	    ImproveMesh (mesh);
+	    if (multithread.terminate)
+	      throw NgException ("Meshing stopped");
+	  }
+	faceindex = 0;
+	return;
+      }
  
-  CheckMeshApproximation (mesh);
+    CheckMeshApproximation (mesh);
 
-  int i, j, k, surfi3;
-  SurfaceElementIndex sei;
+    int i, j, k, surfi3;
+    SurfaceElementIndex sei;
 
-  ARRAY<SurfaceElementIndex> seia;
-  mesh.GetSurfaceElementsOfFace (faceindex, seia);
+    ARRAY<SurfaceElementIndex> seia;
+    mesh.GetSurfaceElementsOfFace (faceindex, seia);
 
-  bool mixed = 0;
-  for (i = 0; i < seia.Size(); i++)
-    if (mesh[seia[i]].GetNP() != 3)
-      {
-	mixed = 1;
-	break;
-      }
+    bool mixed = 0;
+    for (i = 0; i < seia.Size(); i++)
+      if (mesh[seia[i]].GetNP() != 3)
+	{
+	  mixed = 1;
+	  break;
+	}
 
 
-  int loci;
-  double fact;
-  int moveisok;
+    int loci;
+    double fact;
+    int moveisok;
 
-  PointGeomInfo ngi;
-  Point3d origp;
+    PointGeomInfo ngi;
+    Point3d origp;
 
-  Vec3d n1, n2;
-  Vector x(2), xedge(1);
+    Vec3d n1, n2;
+    Vector x(2), xedge(1);
 
-  ARRAY<Point3d, PointIndex::BASE> savepoints(mesh.GetNP());
-  uselocalh = mparam.uselocalh;
+    ARRAY<MeshPoint, PointIndex::BASE> savepoints(mesh.GetNP());
+    uselocalh = mparam.uselocalh;
 
-  ARRAY<int, PointIndex::BASE> nelementsonpoint(mesh.GetNP());
-  nelementsonpoint = 0;
+    ARRAY<int, PointIndex::BASE> nelementsonpoint(mesh.GetNP());
+    nelementsonpoint = 0;
 
-  for (i = 0; i < seia.Size(); i++)
-    {
-      const Element2d & el = mesh[seia[i]];
-      for (j = 0; j < el.GetNP(); j++)
-	nelementsonpoint[el[j]]++;
-    }
+    for (i = 0; i < seia.Size(); i++)
+      {
+	const Element2d & el = mesh[seia[i]];
+	for (j = 0; j < el.GetNP(); j++)
+	  nelementsonpoint[el[j]]++;
+      }
 
 
-  TABLE<SurfaceElementIndex,PointIndex::BASE> elementsonpoint(nelementsonpoint);
-  for (i = 0; i < seia.Size(); i++)
-    {
-      const Element2d & el = mesh[seia[i]];
-      for (j = 0; j < el.GetNP(); j++)
-	elementsonpoint.Add (el[j], seia[i]);
-    }
+    TABLE<SurfaceElementIndex,PointIndex::BASE> elementsonpoint(nelementsonpoint);
+    for (i = 0; i < seia.Size(); i++)
+      {
+	const Element2d & el = mesh[seia[i]];
+	for (j = 0; j < el.GetNP(); j++)
+	  elementsonpoint.Add (el[j], seia[i]);
+      }
 
-  loch = mparam.maxh;
-  locmetricweight = metricweight;
-  meshthis = this;
+    loch = mparam.maxh;
+    locmetricweight = metricweight;
+    meshthis = this;
 
-  Opti2SurfaceMinFunction surfminf(mesh);
-  Opti2EdgeMinFunction edgeminf(mesh);
-  Opti2SurfaceMinFunctionJacobian surfminfj(mesh);
+    Opti2SurfaceMinFunction surfminf(mesh);
+    Opti2EdgeMinFunction edgeminf(mesh);
+    Opti2SurfaceMinFunctionJacobian surfminfj(mesh);
 
-  OptiParameters par;
-  par.maxit_linsearch = 8;
-  par.maxit_bfgs = 5;
+    OptiParameters par;
+    par.maxit_linsearch = 8;
+    par.maxit_bfgs = 5;
 
-  /*
-  if (improveedges)
-    for (i = 1; i <= mesh.GetNP(); i++)
+    /*
+      if (improveedges)
+      for (i = 1; i <= mesh.GetNP(); i++)
       if (mesh.PointType(i) == EDGEPOINT)
-	{
-	  continue;
-	  PrintDot ();
-	  sp1 = mesh.Point(i);
+      {
+      continue;
+      PrintDot ();
+      sp1 = mesh.Point(i);
 	  
-	  locelements.SetSize(0);
-	  locrots.SetSize (0);
-	  lochs.SetSize (0);
-	  surfi = surfi2 = surfi3 = 0;
+      locelements.SetSize(0);
+      locrots.SetSize (0);
+      lochs.SetSize (0);
+      surfi = surfi2 = surfi3 = 0;
 	  
-	  for (j = 0; j < elementsonpoint[i].Size(); j++)
-	    {
-	      sei = elementsonpoint[i][j];
-	      const Element2d * bel = &mesh[sei];
+      for (j = 0; j < elementsonpoint[i].Size(); j++)
+      {
+      sei = elementsonpoint[i][j];
+      const Element2d * bel = &mesh[sei];
 	      
-	      if (!surfi)
-		surfi = mesh.GetFaceDescriptor(bel->GetIndex()).SurfNr();
-	      else if (surfi != mesh.GetFaceDescriptor(bel->GetIndex()).SurfNr())
-		{
-		  if (surfi2 != 0 && surfi2 != 
-		      mesh.GetFaceDescriptor(bel->GetIndex()).SurfNr())
-		    surfi3 = mesh.GetFaceDescriptor(bel->GetIndex()).SurfNr();
-		  else
-		    surfi2 = mesh.GetFaceDescriptor(bel->GetIndex()).SurfNr();
-		}
+      if (!surfi)
+      surfi = mesh.GetFaceDescriptor(bel->GetIndex()).SurfNr();
+      else if (surfi != mesh.GetFaceDescriptor(bel->GetIndex()).SurfNr())
+      {
+      if (surfi2 != 0 && surfi2 != 
+      mesh.GetFaceDescriptor(bel->GetIndex()).SurfNr())
+      surfi3 = mesh.GetFaceDescriptor(bel->GetIndex()).SurfNr();
+      else
+      surfi2 = mesh.GetFaceDescriptor(bel->GetIndex()).SurfNr();
+      }
 	      
-	      locelements.Append (sei);
+      locelements.Append (sei);
 	      
-	      if (bel->PNum(1) == i)
-		locrots.Append (1);
-	      else if (bel->PNum(2) == i)
-		locrots.Append (2);
-	      else
-		locrots.Append (3);
+      if (bel->PNum(1) == i)
+      locrots.Append (1);
+      else if (bel->PNum(2) == i)
+      locrots.Append (2);
+      else
+      locrots.Append (3);
 
-	      if (uselocalh)
-		{
-		  Point3d pmid = Center (mesh.Point(bel->PNum(1)),
-					 mesh.Point(bel->PNum(2)),
-					 mesh.Point(bel->PNum(3)));
-		  lochs.Append (mesh.GetH(pmid));
-		}
-	    }
+      if (uselocalh)
+      {
+      Point3d pmid = Center (mesh.Point(bel->PNum(1)),
+      mesh.Point(bel->PNum(2)),
+      mesh.Point(bel->PNum(3)));
+      lochs.Append (mesh.GetH(pmid));
+      }
+      }
 	  
-	  if (surfi2 && !surfi3)
-	    {
-	      GetNormalVector (surfi, sp1, n1);
-	      GetNormalVector (surfi2, sp1, n2);
-	      t1 = Cross (n1, n2);
+      if (surfi2 && !surfi3)
+      {
+      GetNormalVector (surfi, sp1, n1);
+      GetNormalVector (surfi2, sp1, n2);
+      t1 = Cross (n1, n2);
 	      
-	      xedge = 0;
-	      BFGS (xedge, edgeminf, par, 1e-6);
+      xedge = 0;
+      BFGS (xedge, edgeminf, par, 1e-6);
 	      
-	      mesh.Point(i).X() += xedge.Get(1) * t1.X();
-	      mesh.Point(i).Y() += xedge.Get(1) * t1.Y();
-	      mesh.Point(i).Z() += xedge.Get(1) * t1.Z();
-	      ProjectPoint2 (surfi, surfi2, mesh.Point(i));
-	    }
-	}
-  */
-
-  bool printeddot = 0;
-  char plotchar = '.';
-  int modplot = 1;
-  if (mesh.GetNP() > 1000)
-    {
-      plotchar = '+';
-      modplot = 10;
-    }
-  int cnt = 0;
+      mesh.Point(i).X() += xedge.Get(1) * t1.X();
+      mesh.Point(i).Y() += xedge.Get(1) * t1.Y();
+      mesh.Point(i).Z() += xedge.Get(1) * t1.Z();
+      ProjectPoint2 (surfi, surfi2, mesh.Point(i));
+      }
+      }
+    */
 
-  for (PointIndex pi = PointIndex::BASE; 
-       pi < mesh.GetNP()+PointIndex::BASE; pi++)
 
-    if (mesh.PointType(pi) == SURFACEPOINT)
+    bool printeddot = 0;
+    char plotchar = '.';
+    int modplot = 1;
+    if (mesh.GetNP() > 1000)
       {
-	if (multithread.terminate)
-	  throw NgException ("Meshing stopped");
-	
-	cnt++;
-	if (cnt % modplot == 0 && writestatus)
-	  {
-	    printeddot = 1;
-	    PrintDot (plotchar);
-	  }
-	
-	if (elementsonpoint[pi].Size() == 0)
-	  continue;
+	plotchar = '+';
+	modplot = 10;
+      }
+    if (mesh.GetNP() > 10000)
+      {
+	plotchar = 'o';
+	modplot = 100;
+      }
+    int cnt = 0;
+
+    for (PointIndex pi = PointIndex::BASE; 
+	 pi < mesh.GetNP()+PointIndex::BASE; pi++)
+
+      if (mesh.PointType(pi) == SURFACEPOINT)
+	{
+	  if (multithread.terminate)
+	    throw NgException ("Meshing stopped");
 	
-	sp1 = mesh[pi];
+	  if (pi ==  3679)
+	    (*testout) << " old: " << mesh[pi] << endl;
 
-	Element2d & hel = mesh[elementsonpoint[pi][0]];
 
-	int hpi = 0;
-	for (j = 1; j <= hel.GetNP(); j++)
-	  if (hel.PNum(j) == pi)
+	  cnt++;
+	  if (cnt % modplot == 0 && writestatus)
 	    {
-	      hpi = j;
-	      break;
+	      printeddot = 1;
+	      PrintDot (plotchar);
 	    }
+	
+	  if (elementsonpoint[pi].Size() == 0)
+	    continue;
+	
+	  sp1 = mesh[pi];
+
+	  Element2d & hel = mesh[elementsonpoint[pi][0]];
 
-	gi1 = hel.GeomInfoPi(hpi);
-	SelectSurfaceOfPoint (sp1, gi1);
+	  int hpi = 0;
+	  for (j = 1; j <= hel.GetNP(); j++)
+	    if (hel.PNum(j) == pi)
+	      {
+		hpi = j;
+		break;
+	      }
+
+	  gi1 = hel.GeomInfoPi(hpi);
+	  SelectSurfaceOfPoint (sp1, gi1);
 	  
-	locelements.SetSize(0);
-	locrots.SetSize (0);
-	lochs.SetSize (0);
+	  locelements.SetSize(0);
+	  locrots.SetSize (0);
+	  lochs.SetSize (0);
 	
-	for (j = 0; j < elementsonpoint[pi].Size(); j++)
-	  {
-	    sei = elementsonpoint[pi][j];
-	    const Element2d & bel = mesh[sei];
-	    surfi = mesh.GetFaceDescriptor(bel.GetIndex()).SurfNr();
+	  for (j = 0; j < elementsonpoint[pi].Size(); j++)
+	    {
+	      sei = elementsonpoint[pi][j];
+	      const Element2d & bel = mesh[sei];
+	      surfi = mesh.GetFaceDescriptor(bel.GetIndex()).SurfNr();
 	    
-	    locelements.Append (sei);
+	      locelements.Append (sei);
 	    
-	    for (k = 1; k <= bel.GetNP(); k++)
-	      if (bel.PNum(k) == pi)
+	      for (k = 1; k <= bel.GetNP(); k++)
+		if (bel.PNum(k) == pi)
+		  {
+		    locrots.Append (k);
+		    break;
+		  }
+	      
+	      if (uselocalh)
 		{
-		  locrots.Append (k);
-		  break;
+		  Point3d pmid = Center (mesh[bel[0]], mesh[bel[1]], mesh[bel[2]]);
+		  lochs.Append (mesh.GetH(pmid));
 		}
-	      
-	    if (uselocalh)
-	      {
-		Point3d pmid = Center (mesh[bel[0]], mesh[bel[1]], mesh[bel[2]]);
-		lochs.Append (mesh.GetH(pmid));
-	      }
-	  }
+	    }
 
 	  
-	GetNormalVector (surfi, sp1, gi1, n);
-	n.GetNormal (t1);
-	t2 = Cross (n, t1);
+	  GetNormalVector (surfi, sp1, gi1, n);
+	  n.GetNormal (t1);
+	  t2 = Cross (n, t1);
 	  
-	// save points, and project to tangential plane
-	for (j = 0; j < locelements.Size(); j++)
-	  {
-	    const Element2d & el = mesh[locelements[j]];
-	    for (k = 0; k < el.GetNP(); k++)
-	      savepoints[el[k]] = mesh[el[k]];
-	  }
+	  // save points, and project to tangential plane
+	  for (j = 0; j < locelements.Size(); j++)
+	    {
+	      const Element2d & el = mesh[locelements[j]];
+	      for (k = 0; k < el.GetNP(); k++)
+		savepoints[el[k]] = mesh[el[k]];
+	    }
 
-	for (j = 0; j < locelements.Size(); j++)
-	  {
-	    const Element2d & el = mesh[locelements[j]];
-	    for (k = 0; k < el.GetNP(); k++)
-	      {
-		PointIndex hpi = el[k];
-		double lam = n * (mesh[hpi] - sp1);
-		mesh[hpi] -= lam * n;
-	      }
-	  }
+	  for (j = 0; j < locelements.Size(); j++)
+	    {
+	      const Element2d & el = mesh[locelements[j]];
+	      for (k = 0; k < el.GetNP(); k++)
+		{
+		  PointIndex hpi = el[k];
+		  double lam = n * (mesh[hpi] - sp1);
+		  mesh[hpi] -= lam * n;
+		}
+	    }
 	  
-	x = 0;
+	  x = 0;
 
-	if (mixed)
-	  BFGS (x, surfminfj, par, 1e-6);
-	else
-	  BFGS (x, surfminf, par, 1e-6);
+	  if (mixed)
+	    BFGS (x, surfminfj, par, 1e-6);
+	  else
+	    BFGS (x, surfminf, par, 1e-6);
 
-	origp = mesh[pi];
-	loci = 1;
-	fact = 1;
-	moveisok = 0;
+	  origp = mesh[pi];
+	  loci = 1;
+	  fact = 1;
+	  moveisok = 0;
 
-	// restore other points
-	for (j = 0; j < locelements.Size(); j++)
-	  {
-	    const Element2d & el = mesh[locelements[j]];
-	    for (k = 0; k < el.GetNP(); k++)
-	      {
-		PointIndex hpi = el[k];
-		if (hpi != pi) mesh[hpi] = savepoints[hpi];
-	      }
-	  }
+	  // restore other points
+	  for (j = 0; j < locelements.Size(); j++)
+	    {
+	      const Element2d & el = mesh[locelements[j]];
+	      for (k = 0; k < el.GetNP(); k++)
+		{
+		  PointIndex hpi = el[k];
+		  if (hpi != pi) mesh[hpi] = savepoints[hpi];
+		}
+	    }
 	  
 	  
-	//optimizer loop (if not whole distance is not possible, move only a bit!!!!)
-	while (loci <= 5 && !moveisok)
-	  {
-	    loci ++;
-	    mesh[pi].X() = origp.X() + (x.Get(1) * t1.X() + x.Get(2) * t2.X())*fact;
-	    mesh[pi].Y() = origp.Y() + (x.Get(1) * t1.Y() + x.Get(2) * t2.Y())*fact;
-	    mesh[pi].Z() = origp.Z() + (x.Get(1) * t1.Z() + x.Get(2) * t2.Z())*fact;
-	    fact = fact/2.;
-	    ProjectPoint (surfi, mesh[pi]);
+	  //optimizer loop (if not whole distance is not possible, move only a bit!!!!)
+	  while (loci <= 5 && !moveisok)
+	    {
+	      loci ++;
+	      mesh[pi].X() = origp.X() + (x.Get(1) * t1.X() + x.Get(2) * t2.X())*fact;
+	      mesh[pi].Y() = origp.Y() + (x.Get(1) * t1.Y() + x.Get(2) * t2.Y())*fact;
+	      mesh[pi].Z() = origp.Z() + (x.Get(1) * t1.Z() + x.Get(2) * t2.Z())*fact;
+	      fact = fact/2.;
+
+	      if (pi ==  3679)
+		(*testout) << " before proj: " << mesh[pi] << endl;
+
+	      ProjectPoint (surfi, mesh[pi]);
+
+	      if (pi ==  3679)
+		{
+		  (*testout) << " after proj: " << mesh[pi] << endl;
+		  (*testout) << "surfi =" << surfi << endl;
+		}
 	    
-	    moveisok = CalcPointGeomInfo(surfi, ngi, mesh[pi]); 
-	    // point lies on same chart in stlsurface
+	      moveisok = CalcPointGeomInfo(surfi, ngi, mesh[pi]); 
+	      // point lies on same chart in stlsurface
 	    
-	    if (moveisok)
-	      {
-		for (j = 0; j < locelements.Size(); j++)
-		  mesh[locelements[j]].GeomInfoPi(locrots[j]) = ngi;
-	      }
-	    else
-	      {
-		mesh[pi] = origp;
-	      }
+	      if (moveisok)
+		{
+		  for (j = 0; j < locelements.Size(); j++)
+		    mesh[locelements[j]].GeomInfoPi(locrots[j]) = ngi;
+		}
+	      else
+		{
+		  mesh[pi] = origp;
+		}
 	    
-	  }
-	  //	(*testout) << " new: " << mesh.Point(i) << endl;
-      }
+	    }
+
+	  if (pi ==  3679)
+	    (*testout) << " new: " << mesh[pi] << endl;
+	}
   
-  if (printeddot)
-    PrintDot ('\n');
+
+    if (printeddot)
+      PrintDot ('\n');
   
-  CheckMeshApproximation (mesh);
-  mesh.SetNextTimeStamp();
-}
+    CheckMeshApproximation (mesh);
+    mesh.SetNextTimeStamp();
+  }
 
-void MeshOptimize2d :: GetNormalVector(INDEX /* surfind */, const Point3d & p, Vec3d & nv) const
-{
-  nv = Vec3d (0, 0, 1);
-}
+  void MeshOptimize2d :: GetNormalVector(INDEX /* surfind */, const Point3d & p, Vec3d & nv) const
+  {
+    nv = Vec3d (0, 0, 1);
+  }
 
-void MeshOptimize2d :: GetNormalVector(INDEX surfind, const Point3d & p, PointGeomInfo & gi, Vec3d & n) const
-{
-  GetNormalVector (surfind, p, n);
-}
+  void MeshOptimize2d :: GetNormalVector(INDEX surfind, const Point3d & p, PointGeomInfo & gi, Vec3d & n) const
+  {
+    GetNormalVector (surfind, p, n);
+  }
 
 }
diff --git a/Netgen/libsrc/meshing/smoothing3.cpp b/Netgen/libsrc/meshing/smoothing3.cpp
index 6fcb2b8e70..61c1910884 100644
--- a/Netgen/libsrc/meshing/smoothing3.cpp
+++ b/Netgen/libsrc/meshing/smoothing3.cpp
@@ -24,15 +24,15 @@ namespace netgen
   {
     int j;
     double badness = 0;
-    Point3d pp(vp.Get(1), vp.Get(2), vp.Get(3));
+    Point3d pp(vp(0), vp(1), vp(2));
 
     for (j = 0; j < faces.Size(); j++)
       {
 	const INDEX_3 & el = faces[j];
 
-	double bad = CalcTetBadness (points.Get(el.I1()), 
-				     points.Get(el.I3()), 
-				     points.Get(el.I2()), 
+	double bad = CalcTetBadness (points[el.I1()], 
+				     points[el.I3()], 
+				     points[el.I2()], 
 				     pp, 0);
 	badness += bad;
       }
@@ -42,7 +42,8 @@ namespace netgen
 
 
 
-  double PointFunction1 :: FuncDeriv (const Vector & x, const Vector & dir, double & deriv) const
+  double PointFunction1 :: 
+  FuncDeriv (const Vector & x, const Vector & dir, double & deriv) const
   {
     static Vector hx(3);
     static double eps = 1e-6;
@@ -127,9 +128,9 @@ namespace netgen
   
     for (i = 1; i <= nf; i++)
       {
-	const Point3d & p1 = points.Get(faces.Get(i).I1());
-	const Point3d & p2 = points.Get(faces.Get(i).I2());
-	const Point3d & p3 = points.Get(faces.Get(i).I3());
+	const Point3d & p1 = points[faces.Get(i).I1()];
+	const Point3d & p2 = points[faces.Get(i).I2()];
+	const Point3d & p3 = points[faces.Get(i).I3()];
 	Vec3d v1 (p1, p2);
 	Vec3d v2 (p1, p3);
 	Vec3d n;
@@ -229,15 +230,15 @@ namespace netgen
   public:
     Mesh::T_POINTS & points;
     const Mesh::T_VOLELEMENTS & elements;
-    TABLE<INDEX> elementsonpoint;
-    int actpind;
+    TABLE<INDEX,PointIndex::BASE> elementsonpoint;
+    PointIndex actpind;
     double h;
   
   public:
     PointFunction (Mesh::T_POINTS & apoints, 
 		   const Mesh::T_VOLELEMENTS & aelements);
   
-    virtual void SetPointIndex (int aactpind);
+    virtual void SetPointIndex (PointIndex aactpind);
     void SetLocalH (double ah) { h = ah; }
     double GetLocalH () const { return h; }
     virtual double PointFunctionValue (const Point3d & pp) const;
@@ -259,11 +260,11 @@ namespace netgen
       {
 	if (elements.Get(i).NP() == 4)
 	  for (j = 1; j <= elements.Get(i).NP(); j++)
-	    elementsonpoint.Add1 (elements.Get(i).PNum(j), i);  
+	    elementsonpoint.Add (elements.Get(i).PNum(j), i);  
       }
   }
 
-  void PointFunction :: SetPointIndex (int aactpind)
+  void PointFunction :: SetPointIndex (PointIndex aactpind)
   {
     actpind = aactpind; 
   }  
@@ -279,20 +280,20 @@ namespace netgen
 
     badness = 0;
 
-    hp = points.Elem(actpind);
-    points.Elem(actpind) = pp;
+    hp = points[actpind];
+    points[actpind] = pp;
 
-    for (j = 1; j <= elementsonpoint.EntrySize(actpind); j++)
+    for (j = 0; j < elementsonpoint[actpind].Size(); j++)
       {
-	eli = elementsonpoint.Get(actpind, j);
+	eli = elementsonpoint[actpind][j];
 	el = &elements.Get(eli);
-	badness += CalcTetBadness (points.Get(el->PNum(1)), 
-				   points.Get(el->PNum(2)), 
-				   points.Get(el->PNum(3)), 
-				   points.Get(el->PNum(4)), -1);
+	badness += CalcTetBadness (points[el->PNum(1)], 
+				   points[el->PNum(2)], 
+				   points[el->PNum(3)], 
+				   points[el->PNum(4)], -1);
       }
   
-    points.Elem(actpind) = hp; 
+    points[actpind] = hp; 
     return badness;
   }
 
@@ -338,25 +339,25 @@ namespace netgen
 
     //  badness = 0;
 
-    hp = points.Elem(actpind);
-    points.Elem(actpind) = pp;
+    hp = points[actpind];
+    points[actpind] = pp;
 
-    for (j = 1; j <= elementsonpoint.EntrySize(actpind); j++)
+    for (j = 0; j < elementsonpoint[actpind].Size(); j++)
       {
-	eli = elementsonpoint.Get(actpind, j);
+	eli = elementsonpoint[actpind][j];
 	const Element & el = elements.Get(eli);
 
 	for (k = 1; k <= 4; k++)
 	  if (el.PNum(k) == actpind)
 	    {
-	      CalcTetBadnessGrad (points.Get(el.PNum(1)), 
-				  points.Get(el.PNum(2)), 
-				  points.Get(el.PNum(3)), 
-				  points.Get(el.PNum(4)), -1, k, vgradi);
+	      CalcTetBadnessGrad (points[el.PNum(1)], 
+				  points[el.PNum(2)], 
+				  points[el.PNum(3)], 
+				  points[el.PNum(4)], -1, k, vgradi);
 	      vgrad += vgradi;
 	    }
       }
-    points.Elem(actpind) = hp; 
+    points[actpind] = hp; 
 
     for (j = 1; j <= 3; j++)
       grad.Elem(j) = vgrad.X(j);
@@ -382,28 +383,28 @@ namespace netgen
 
     //  badness = 0;
 
-    hp = points.Elem(actpind);
-    points.Elem(actpind) = pp;
+    hp = points[actpind];
+    points[actpind] = pp;
     f = 0;
 
-    for (j = 1; j <= elementsonpoint.EntrySize(actpind); j++)
+    for (j = 0; j < elementsonpoint[actpind].Size(); j++)
       {
-	eli = elementsonpoint.Get(actpind, j);
+	eli = elementsonpoint[actpind][j];
 	const Element & el = elements.Get(eli);
 
 	for (k = 1; k <= 4; k++)
 	  if (el.PNum(k) == actpind)
 	    {
-	      f += CalcTetBadnessGrad (points.Get(el.PNum(1)), 
-				       points.Get(el.PNum(2)), 
-				       points.Get(el.PNum(3)), 
-				       points.Get(el.PNum(4)), -1, k, vgradi);
+	      f += CalcTetBadnessGrad (points[el.PNum(1)], 
+				       points[el.PNum(2)], 
+				       points[el.PNum(3)], 
+				       points[el.PNum(4)], -1, k, vgradi);
 
 	      vgrad += vgradi;
 	    }
       }
 
-    points.Elem(actpind) = hp; 
+    points[actpind] = hp; 
     deriv = dir * vgrad;
     return f;
   }
@@ -415,10 +416,10 @@ namespace netgen
     // try point movement 
     ARRAY<Element2d> faces;
   
-    for (j = 1; j <= elementsonpoint.EntrySize(actpind); j++)
+    for (j = 0; j < elementsonpoint[actpind].Size(); j++)
       {
 	const Element & el = 
-	  elements.Get(elementsonpoint.Get (actpind, j));
+	  elements.Get(elementsonpoint[actpind][j]);
       
 	for (k = 1; k <= 4; k++)
 	  if (el.PNum(k) == actpind)
@@ -435,12 +436,10 @@ namespace netgen
     if (hi)
       {
 	cout << "inner point found" << endl;
-	points.Elem(actpind) = hp;
+	points[actpind] = hp;
       }
     else
       cout << "no inner point found" << endl;
-
-
   
     return hi;
   }
@@ -456,7 +455,7 @@ namespace netgen
   public:
     CheapPointFunction (Mesh::T_POINTS & apoints, 
 			const Mesh::T_VOLELEMENTS & aelements);
-    virtual void SetPointIndex (int aactpind);
+    virtual void SetPointIndex (PointIndex aactpind);
     virtual double PointFunctionValue (const Point3d & pp) const;
     virtual double PointFunctionValueGrad (const Point3d & pp, Vector & grad) const;
   };
@@ -470,23 +469,23 @@ namespace netgen
   }
 
 
-  void CheapPointFunction :: SetPointIndex (int aactpind)
+  void CheapPointFunction :: SetPointIndex (PointIndex aactpind)
   {
     actpind = aactpind; 
 
-    int n = elementsonpoint.EntrySize(actpind);
+    int n = elementsonpoint[actpind].Size();
     int i, j;
     int pi1, pi2, pi3;
 
     m.SetSize (n, 4);
 
-    for (i = 1; i <= n; i++)
+    for (i = 0; i < n; i++)
       {
 	pi1 = 0;
 	pi2 = 0;
 	pi3 = 0;
 
-	const Element & el = elements.Get (elementsonpoint.Get(actpind, i));
+	const Element & el = elements.Get (elementsonpoint[actpind][i]);
 	for (j = 1; j <= 4; j++)
 	  if (el.PNum(j) != actpind)
 	    {
@@ -495,14 +494,14 @@ namespace netgen
 	      pi1 = el.PNum(j);
 	    }
 
-	const Point3d & p1 = points.Get(pi1);
-	Vec3d v1 (p1, points.Get(pi2));
-	Vec3d v2 (p1, points.Get(pi3));
+	const Point3d & p1 = points[pi1];
+	Vec3d v1 (p1, points[pi2]);
+	Vec3d v2 (p1, points[pi3]);
 	Vec3d n;
 	Cross (v1, v2, n);
 	n /= n.Length();
 
-	Vec3d v (p1, points.Get(actpind));
+	Vec3d v (p1, points[actpind]);
 	double c = v * n;
       
 	if (c < 0)
@@ -976,13 +975,13 @@ public:
   Mesh::T_POINTS & points;
   const Mesh::T_VOLELEMENTS & elements;
   TABLE<INDEX> elementsonpoint;
-  int actpind;
+  PointIndex actpind;
   
 public:
   JacobianPointFunction (Mesh::T_POINTS & apoints, 
 			 const Mesh::T_VOLELEMENTS & aelements);
   
-  virtual void SetPointIndex (int aactpind);
+  virtual void SetPointIndex (PointIndex aactpind);
   virtual double Func (const Vector & x) const;
   virtual double FuncGrad (const Vector & x, Vector & g) const;
   virtual double FuncDeriv (const Vector & x, const Vector & dir, double & deriv) const;
@@ -1004,7 +1003,7 @@ JacobianPointFunction (Mesh::T_POINTS & apoints,
     }
 }
 
-void JacobianPointFunction :: SetPointIndex (int aactpind)
+void JacobianPointFunction :: SetPointIndex (PointIndex aactpind)
 {
   actpind = aactpind; 
 }  
@@ -1320,40 +1319,39 @@ void Mesh :: ImproveMesh (const CSGeometry & geometry, OPTIMIZEGOAL goal)
 void Mesh :: ImproveMesh (OPTIMIZEGOAL goal)
 {
   int typ = 1;
-  int i, j;
+  int j;
   
-  (*testout) << "Improve Mesh2" << "\n";
-  PrintMessage (3, "ImproveMesh2");
+  (*testout) << "Improve Mesh" << "\n";
+  PrintMessage (3, "ImproveMesh");
 
   int np = GetNP();
   int ne = GetNE();
 
 
-  ARRAY<double> perrs(np);
-  for (i = 1; i <= np; i++)
-    perrs.Elem(i) = 1;
+  ARRAY<double,PointIndex::BASE> perrs(np);
+  perrs = 1.0;
 
   double bad1 = 0;
   double badmax = 0;
 
   if (goal == OPT_QUALITY)
     {
-      for (i = 1; i <= ne; i++)
+      for (int i = 1; i <= ne; i++)
 	{
 	  const Element & el = VolumeElement(i);
 	  if (el.GetType() != TET)
 	    continue;
 	  
 	  double hbad = CalcBad (points, el, 0);
-	  for (j = 1; j <= 4; j++)
-	    perrs.Elem(el.PNum(j)) += hbad;
+	  for (j = 0; j < 4; j++)
+	    perrs[el[j]] += hbad;
 	  
 	  bad1 += hbad;
 	}
       
-      for (i = 1; i <= np; i++)
-	if (perrs.Get(i) > badmax)
-	  badmax = perrs.Get(i);
+      for (PointIndex i = PointIndex::BASE; i < np+PointIndex::BASE; i++)
+	if (perrs[i] > badmax) 
+	  badmax = perrs[i];
       badmax = 0;
     }
 
@@ -1390,35 +1388,26 @@ void Mesh :: ImproveMesh (OPTIMIZEGOAL goal)
   char * savetask = multithread.task;
   multithread.task = "Smooth Mesh";
   
-  for (i = 1; i <= points.Size(); i++)
-    if (PointType(i) == INNERPOINT && perrs.Get(i) > 0.01 * badmax)
+  for (PointIndex i = PointIndex::BASE; 
+       i < points.Size()+PointIndex::BASE; i++)
+    if (PointType(i) == INNERPOINT && perrs[i] > 0.01 * badmax)
       {
 	if (multithread.terminate)
 	  throw NgException ("Meshing stopped");
-	/*
-	if (multithread.terminate)
-	  break;
-	*/
 
-	multithread.percent = 100.0 * i / points.Size();
+	multithread.percent = 100.0 * (i+1-PointIndex::BASE) / points.Size();
 
 	if (points.Size() < 1000)
 	  PrintDot ();
 	else
-	  if (i % 10 == 0)
+	  if ( (i+1-PointIndex::BASE) % 10 == 0)
 	    PrintDot ('+');
 
-	//	if (uselocalh)
-	{
-	  double lh = GetH(points.Get(i));
-	  pf->SetLocalH (lh);
-	  par.typx = lh;
-	  //	    (*mycout) << "lh = " << lh << "\n";
-	}
+	double lh = GetH(points[i]);
+	pf->SetLocalH (lh);
+	par.typx = lh;
 
-	//    if (elementsonpoint.EntrySize(i) == 0) continue;
-	
-	freeminf.SetPoint (points.Elem(i));
+	freeminf.SetPoint (points[i]);
 	pf->SetPointIndex (i);
 
 	x = 0;
@@ -1429,7 +1418,7 @@ void Mesh :: ImproveMesh (OPTIMIZEGOAL goal)
 	  {
 	    pok = pf->MovePointToInner ();
 
-	    freeminf.SetPoint (points.Elem(i));
+	    freeminf.SetPoint (points[i]);
 	    pf->SetPointIndex (i);
 	  }
 
@@ -1437,9 +1426,9 @@ void Mesh :: ImproveMesh (OPTIMIZEGOAL goal)
 	  {
 	    BFGS (x, freeminf, par);
 	    
-	    points.Elem(i).X() += x.Get(1);
-	    points.Elem(i).Y() += x.Get(2);
-	    points.Elem(i).Z() += x.Get(3);
+	    points[i].X() += x.Get(1);
+	    points[i].Y() += x.Get(2);
+	    points[i].Z() += x.Get(3);
 	  }
       }
   PrintDot ('\n');
diff --git a/Netgen/libsrc/meshing/topology.cpp b/Netgen/libsrc/meshing/topology.cpp
index 9b91a464dc..9c0e79d531 100644
--- a/Netgen/libsrc/meshing/topology.cpp
+++ b/Netgen/libsrc/meshing/topology.cpp
@@ -26,6 +26,13 @@ MeshTopology ::  MeshTopology (const Mesh & amesh)
   face2surfel.SetName ("face2surfel");
 }
 
+MeshTopology :: ~MeshTopology ()
+{
+  delete vert2element;
+  delete vert2surfelement;
+  delete vert2segment;
+}
+
 void MeshTopology :: Update()
 {
   int i, j, k;
@@ -51,7 +58,7 @@ void MeshTopology :: Update()
   delete vert2element;
   delete vert2surfelement;
   delete vert2segment;
-  ARRAY<int> cnt(nv);
+  ARRAY<int,PointIndex::BASE> cnt(nv);
   ARRAY<int> vnums;
 
   (*testout) << "tables deleted" << endl;
@@ -69,9 +76,10 @@ void MeshTopology :: Update()
       const Element & el = mesh.VolumeElement(i);
       int nelv = el.GetNV();
       for (j = 1; j <= nelv; j++)
-	cnt.Elem(el.PNum(j))++;
+	cnt[el.PNum(j)]++;
     }
-  vert2element = new TABLE<int> (cnt);
+
+  vert2element = new TABLE<int,PointIndex::BASE> (cnt);
   for (i = 1; i <= ne; i++)
     {
       const Element & el = mesh.VolumeElement(i);
@@ -81,21 +89,21 @@ void MeshTopology :: Update()
     }
 
   cnt = 0;
-  for (i = 1; i <= nse; i++)
+  for (SurfaceElementIndex sei = 0; sei < nse; sei++)
     {
-      const Element2d & el = mesh.SurfaceElement(i);
+      const Element2d & el = mesh[sei];
       int nelv = el.GetNV();
-      for (j = 1; j <= nelv; j++)
-	cnt.Elem(el.PNum(j))++;
+      for (j = 0; j < nelv; j++)
+	cnt[el[j]]++;
     }
 
-  vert2surfelement = new TABLE<int> (cnt);
-  for (i = 1; i <= nse; i++)
+  vert2surfelement = new TABLE<int,PointIndex::BASE> (cnt);
+  for (SurfaceElementIndex sei = 0; sei < nse; sei++)
     {
-      const Element2d & el = mesh.SurfaceElement(i);
+      const Element2d & el = mesh[sei];
       int nelv = el.GetNV();
-      for (j = 1; j <= nelv; j++)
-	vert2surfelement->AddSave (el.PNum(j), i);
+      for (j = 0; j < nelv; j++)
+	vert2surfelement->AddSave (el[j], sei+1);
     }
 
 
@@ -103,11 +111,11 @@ void MeshTopology :: Update()
   for (i = 1; i <= nseg; i++)
     {
       const Segment & seg = mesh.LineSegment(i);
-      cnt.Elem (seg.p1)++;
-      cnt.Elem (seg.p2)++;
+      cnt[seg.p1]++;
+      cnt[seg.p2]++;
     }
 
-  vert2segment = new TABLE<int> (cnt);
+  vert2segment = new TABLE<int,PointIndex::BASE> (cnt);
   for (i = 1; i <= nseg; i++)
     {
       const Segment & seg = mesh.LineSegment(i);
@@ -117,8 +125,6 @@ void MeshTopology :: Update()
 
 
 
-  (*testout) << "before buildedges" << endl;
-
 
   if (buildedges)
     {
@@ -137,11 +143,11 @@ void MeshTopology :: Update()
 
       // keep existing edges
       cnt = 0;
-      for (i = 1; i <= edge2vert.Size(); i++)
-	cnt.Elem(edge2vert.Get(i)[0])++;
-      TABLE<int> vert2edge (cnt);
-      for (i = 1; i <= edge2vert.Size(); i++)
-	vert2edge.AddSave (edge2vert.Get(i)[0], i);
+      for (i = 0; i < edge2vert.Size(); i++)
+	cnt[edge2vert[i][0]]++;
+      TABLE<int,PointIndex::BASE> vert2edge (cnt);
+      for (i = 0; i < edge2vert.Size(); i++)
+	vert2edge.AddSave (edge2vert[i][0], i+1);
 
       // ensure all coarse grid and intermediate level edges
       cnt = 0;
@@ -154,21 +160,21 @@ void MeshTopology :: Update()
 	  if (pa[0] > 0)
 	    cnt.Elem(pa[0])++;
 	}
-      TABLE<int> vert2vertcoarse (cnt);
+      TABLE<int,PointIndex::BASE> vert2vertcoarse (cnt);
       for (i = 1; i <= mesh.mlbetweennodes.Size(); i++)
 	{
 	  int pa[2];
 	  pa[0] = mesh.mlbetweennodes.Get(i).I1();
 	  pa[1] = mesh.mlbetweennodes.Get(i).I2();
-	  if (pa[0] > pa[1]) Swap (pa[0], pa[1]);
+	  if (pa[0] > pa[1]) swap (pa[0], pa[1]);
 	  if (pa[0] > 0)
-	    vert2vertcoarse.AddSave (pa[0], pa[1]);
+	    vert2vertcoarse.AddSave1 (pa[0], pa[1]);
 	}
 
 
-      ARRAY<int> edgenr(nv), edgeflag(nv);
-      for (i = 1; i <= nv; i++)
-	edgeflag.Elem(i) = 0;
+      ARRAY<int,PointIndex::BASE> edgenr(nv), edgeflag(nv);
+      for (i = PointIndex::BASE; i < nv+PointIndex::BASE; i++)
+	edgeflag[i] = 0;
       ned = edge2vert.Size();
       ARRAY<INDEX_3> missing;
 
@@ -207,7 +213,7 @@ void MeshTopology :: Update()
 			       el.PNum(eledges[k][1]));
 	      
 		  int edgedir = (edge.I1() > edge.I2());
-		  if (edgedir) Swap (edge.I1(), edge.I2());
+		  if (edgedir) swap (edge.I1(), edge.I2());
 	     
 		  if (edge.I1() != i)
 		    continue;
@@ -239,7 +245,7 @@ void MeshTopology :: Update()
 			       el.PNum(eledges[k][1]));
 	      
 		  int edgedir = (edge.I1() > edge.I2());
-		  if (edgedir) Swap (edge.I1(), edge.I2());
+		  if (edgedir) swap (edge.I1(), edge.I2());
 	     
 		  if (edge.I1() != i)
 		    continue;
@@ -265,7 +271,7 @@ void MeshTopology :: Update()
 	      INDEX_2 edge(el.p1, el.p2);
 	      
 	      int edgedir = (edge.I1() > edge.I2());
-	      if (edgedir) Swap (edge.I1(), edge.I2());
+	      if (edgedir) swap (edge.I1(), edge.I2());
 	      
 	      if (edge.I1() != i)
 		continue;
@@ -298,7 +304,7 @@ void MeshTopology :: Update()
 			   el.PNum(eledges[k][1]));
 	  
 	      int edgedir = (edge.I1() > edge.I2());
-	      if (edgedir) Swap (edge.I1(), edge.I2());
+	      if (edgedir) swap (edge.I1(), edge.I2());
 
 	      int edgenum = abs (edges.Elem(i)[k]);
 
@@ -319,7 +325,7 @@ void MeshTopology :: Update()
 			   el.PNum(eledges[k][1]));
 	  
 	      int edgedir = (edge.I1() > edge.I2());
-	      if (edgedir) Swap (edge.I1(), edge.I2());
+	      if (edgedir) swap (edge.I1(), edge.I2());
 
 	      int edgenum = abs (surfedges.Elem(i)[k]);
 
@@ -334,7 +340,7 @@ void MeshTopology :: Update()
       
 	  INDEX_2 edge(el.p1, el.p2);
 	  int edgedir = (edge.I1() > edge.I2());
-	  if (edgedir) Swap (edge.I1(), edge.I2());
+	  if (edgedir) swap (edge.I1(), edge.I2());
 	  
 	  int edgenum = abs (segedges.Elem(i));
 	  
@@ -363,6 +369,9 @@ void MeshTopology :: Update()
       */
     }
 
+
+  //  cout << "build edges done" << endl;
+
 #ifdef OLD
   if (buildedges == 2)
     { // old style with hash-table
@@ -399,7 +408,7 @@ void MeshTopology :: Update()
 			   el.PNum(eledges[j][1]));
 	      
 	      edgedir = (edge.I1() > edge.I2());
-	      if (edgedir) Swap (edge.I1(), edge.I2());
+	      if (edgedir) swap (edge.I1(), edge.I2());
 	      
 	      if (vert2edge.Used (edge))
 		edgenum = vert2edge.Get(edge);
@@ -439,7 +448,7 @@ void MeshTopology :: Update()
 			   el.PNum(eledges[j][1]));
 	      
 	      edgedir = (edge.I1() > edge.I2());
-	      if (edgedir) Swap (edge.I1(), edge.I2());
+	      if (edgedir) swap (edge.I1(), edge.I2());
 	      
 	      if (vert2edge.Used (edge))
 		edgenum = vert2edge.Get(edge);
@@ -484,7 +493,7 @@ void MeshTopology :: Update()
 	    {
 	      INDEX_2 edge(parents[0], parents[1]);
 	      int edgedir = (edge.I1() > edge.I2());
-	      if (edgedir) Swap (edge.I1(), edge.I2());
+	      if (edgedir) swap (edge.I1(), edge.I2());
 	      
 	      if (!vert2edge.Used (edge))
 		{
@@ -519,7 +528,7 @@ void MeshTopology :: Update()
 	      INDEX_2 edge(seg.p1, seg.p2);
 	      int edgenum;
 	      int edgedir = (edge.I1() > edge.I2());
-	      if (edgedir) Swap (edge.I1(), edge.I2());
+	      if (edgedir) swap (edge.I1(), edge.I2());
 	      
 	      if (vert2edge.Used (edge))
 		edgenum = vert2edge.Get(edge);
@@ -593,17 +602,17 @@ void MeshTopology :: Update()
 		facedir = 0;
 		if (face.I1() > face.I2())
 		  {
-		    Swap (face.I1(), face.I2());
+		    swap (face.I1(), face.I2());
 		    facedir += 1;
 		  }
 		if (face.I2() > face.I3())
 		  {
-		    Swap (face.I2(), face.I3());
+		    swap (face.I2(), face.I3());
 		    facedir += 2;
 		  }
 		if (face.I1() > face.I2())
 		  {
-		    Swap (face.I1(), face.I2());
+		    swap (face.I1(), face.I2());
 		    facedir += 4;
 		  }
 		
@@ -644,20 +653,20 @@ void MeshTopology :: Update()
 		    min2 (face4.I4(), face4.I3())) 
 		  {  // z - flip
 		    facedir += 1; 
-		    Swap (face4.I1(), face4.I4());
-		    Swap (face4.I2(), face4.I3());
+		    swap (face4.I1(), face4.I4());
+		    swap (face4.I2(), face4.I3());
 		  }
 		if (min2 (face4.I1(), face4.I4()) >
 		    min2 (face4.I2(), face4.I3())) 
 		  {  // x - flip
 		    facedir += 2; 
-		    Swap (face4.I1(), face4.I2());
-		    Swap (face4.I3(), face4.I4());
+		    swap (face4.I1(), face4.I2());
+		    swap (face4.I3(), face4.I4());
 		  }
 		if (face4.I2() > face4.I4())
 		  {  // diagonal flip
 		    facedir += 4; 
-		    Swap (face4.I2(), face4.I4());
+		    swap (face4.I2(), face4.I4());
 		  }
 		//		face4.Sort();
 		
@@ -712,17 +721,17 @@ void MeshTopology :: Update()
 	      facedir = 0;
 	      if (face.I1() > face.I2())
 		{
-		  Swap (face.I1(), face.I2());
+		  swap (face.I1(), face.I2());
 		  facedir += 1;
 		}
 	      if (face.I2() > face.I3())
 		{
-		  Swap (face.I2(), face.I3());
+		  swap (face.I2(), face.I3());
 		  facedir += 2;
 		}
 	      if (face.I1() > face.I2())
 		{
-		  Swap (face.I1(), face.I2());
+		  swap (face.I1(), face.I2());
 		  facedir += 4;
 		}
 	      
@@ -764,20 +773,20 @@ void MeshTopology :: Update()
 		  min2 (face4.I4(), face4.I3())) 
 		{  // z - orientation
 		  facedir += 1; 
-		  Swap (face4.I1(), face4.I4());
-		  Swap (face4.I2(), face4.I3());
+		  swap (face4.I1(), face4.I4());
+		  swap (face4.I2(), face4.I3());
 		}
 	      if (min2 (face4.I1(), face4.I4()) >
 		  min2 (face4.I2(), face4.I3())) 
 		{  // x - orientation
 		  facedir += 2; 
-		  Swap (face4.I1(), face4.I2());
-		  Swap (face4.I3(), face4.I4());
+		  swap (face4.I1(), face4.I2());
+		  swap (face4.I3(), face4.I4());
 		}
 	      if (face4.I2() > face4.I4())
 		{ 
 		  facedir += 4; 
-		  Swap (face4.I2(), face4.I4());
+		  swap (face4.I2(), face4.I4());
 		}
 	      
 	      INDEX_3 face(face4.I1(), face4.I2(), face4.I3());
@@ -834,6 +843,45 @@ void MeshTopology :: Update()
 	cout << "hashtable: ";
 	vert2face.PrintMemInfo(cout);
       */
+
+
+      ARRAY<char> face_els(nfa), face_surfels(nfa);
+      face_els = 0;
+      face_surfels = 0;
+      ARRAY<int> hfaces;
+      for (i = 1; i <= ne; i++)
+	{
+	  GetElementFaces (i, hfaces);
+	  for (j = 0; j < hfaces.Size(); j++)
+	    face_els[hfaces[j]-1]++;
+	}
+      for (i = 1; i <= nse; i++)
+	face_surfels[GetSurfaceElementFace (i)-1]++;
+
+      if (ne)
+	{
+	  int cnt_err = 0;
+	  for (i = 0; i < nfa; i++)
+	    {
+	      (*testout) << "face " << i << " has " << int(face_els[i]) << " els, " 
+			 << int(face_surfels[i]) << " surfels, tot = "
+			 << face_els[i] + face_surfels[i] << endl; 
+	      
+	      if (face_els[i] + face_surfels[i] == 1)
+		{
+		  cnt_err++;
+		  (*testout) << "illegal face : " << i << endl;
+		  (*testout) << "points = " << face2vert[i] << endl;
+		  (*testout) << "pos = ";
+		  for (int j = 0; j < 4; j++)
+		    if (face2vert[i].I(j+1) >= 1)
+		      (*testout) << mesh[(PointIndex)face2vert[i].I(j+1)] << " ";
+		  (*testout) << endl;
+		}
+	    }
+	  if (cnt_err)
+	    cout << cnt_err << " elements are not matching !!!" << endl;
+	}
     }
   
   /*
@@ -868,6 +916,7 @@ int MeshTopology :: GetNVertices (ELEMENT_TYPE et)
 
     case QUAD:
     case QUAD6:
+    case QUAD8:
       return 4;
 
     case TET:
@@ -904,6 +953,7 @@ int MeshTopology :: GetNEdges (ELEMENT_TYPE et)
 
     case QUAD:
     case QUAD6:
+    case QUAD8:
       return 4;
 
     case TET:
@@ -941,6 +991,7 @@ int MeshTopology :: GetNFaces (ELEMENT_TYPE et)
 
     case QUAD:
     case QUAD6:
+    case QUAD8:
       return 1;
 
     case TET:
@@ -1032,6 +1083,7 @@ const Point3d * MeshTopology :: GetVertices (ELEMENT_TYPE et)
 
     case QUAD:
     case QUAD6:
+    case QUAD8:
       return quad_points;
 
     case TET:
@@ -1130,6 +1182,7 @@ const ELEMENT_EDGE * MeshTopology :: GetEdges (ELEMENT_TYPE et)
 
     case QUAD:
     case QUAD6:
+    case QUAD8:
       return quad_edges;
 
     case TET:
@@ -1148,7 +1201,7 @@ const ELEMENT_EDGE * MeshTopology :: GetEdges (ELEMENT_TYPE et)
     default:
       cerr << "Ng_ME_GetEdges, illegal element type " << et << endl;
     }
-  return 0;  
+   return 0;  
 }
 
 
@@ -1161,13 +1214,13 @@ const ELEMENT_FACE * MeshTopology :: GetFaces (ELEMENT_TYPE et)
 
   static int tet_faces[4][4] =
     { { 4, 2, 3, 0 },
-      { 4, 1, 3, 0 },
+      { 4, 3, 1, 0 },
       { 4, 1, 2, 0 },
-      { 1, 2, 3, 0 } };
+      { 1, 3, 2, 0 } };
   
   static int prism_faces[5][4] =
     {
-      { 1, 2, 3, 0 },
+      { 1, 3, 2, 0 },
       { 4, 5, 6, 0 },
       { 3, 1, 4, 6 },
       { 1, 2, 5, 4 },
@@ -1180,12 +1233,12 @@ const ELEMENT_FACE * MeshTopology :: GetFaces (ELEMENT_TYPE et)
       { 2, 3, 5, 0 },
       { 3, 4, 5, 0 },
       { 4, 1, 5, 0 },
-      { 1, 2, 3, 4 } 
+      { 1, 4, 3, 2 } 
     };
 
   static int hex_faces[6][4] =
     {
-      { 1, 2, 3, 4 },
+      { 1, 4, 3, 2 },
       { 5, 6, 7, 8 },
       { 1, 2, 6, 5 },
       { 2, 3, 7, 6 },
@@ -1203,6 +1256,7 @@ const ELEMENT_FACE * MeshTopology :: GetFaces (ELEMENT_TYPE et)
 
     case QUAD:
     case QUAD6:
+    case QUAD8:
       return quad_faces;
 
 
@@ -1375,10 +1429,9 @@ int MeshTopology :: GetSurfaceElementFace (int elnr) const
 void MeshTopology :: 
 GetSurfaceElementEdgeOrientations (int elnr, ARRAY<int> & eorient) const
 {
-  int i;
   int ned = GetNEdges (mesh.SurfaceElement(elnr).GetType());
   eorient.SetSize (ned);
-  for (i = 1; i <= ned; i++)
+  for (int i = 1; i <= ned; i++)
     eorient.Elem(i) = (surfedges.Get(elnr)[i-1] > 0) ? 1 : -1;
 }
 
@@ -1448,23 +1501,22 @@ void MeshTopology :: GetEdgeVertices (int ednr, int & v1, int & v2) const
 void MeshTopology :: GetFaceEdges (int fnr, ARRAY<int> & edges) const
 {
   ArrayMem<int,4> pi(4);
-  // ArrayMem<int,50> els;
   ArrayMem<int,12> eledges;
 
   edges.SetSize (0);
   GetFaceVertices (fnr, pi);
-  // GetVertexElements (pi[0], els);
+
+  //  GetVertexElements (pi[0], els);
   FlatArray<int> els = GetVertexElements (pi[0]);
 
   // find one element having all vertices of the face
-  int i, j, k;
-  for (i = 0; i < els.Size(); i++)
+  for (int i = 0; i < els.Size(); i++)
     {
       const Element & el = mesh.VolumeElement(els[i]);
-      
+
       int cntv = 0;
-      for (j = 0; j < el.GetNV(); j++)
-	for (k = 0; k < pi.Size(); k++)
+      for (int j = 0; j < el.GetNV(); j++)
+	for (int k = 0; k < pi.Size(); k++)
 	  if (el[j] == pi[k])
 	    cntv++;
 
@@ -1472,13 +1524,13 @@ void MeshTopology :: GetFaceEdges (int fnr, ARRAY<int> & edges) const
 	{
 	  GetElementEdges (els[i], eledges);
 	  
-	  for (j = 0; j < eledges.Size(); j++)
+	  for (int j = 0; j < eledges.Size(); j++)
 	    {
 	      int vi1, vi2;
 	      GetEdgeVertices (eledges[j], vi1, vi2);
 	      bool has1 = 0;
 	      bool has2 = 0;
-	      for (k = 0; k < pi.Size(); k++)
+	      for (int k = 0; k < pi.Size(); k++)
 		{
 		  if (vi1 == pi[k]) has1 = 1;
 		  if (vi2 == pi[k]) has2 = 1;
@@ -1515,9 +1567,29 @@ void MeshTopology :: GetVertexElements (int vnr, ARRAY<int> & elements) const
 FlatArray<int> MeshTopology :: GetVertexElements (int vnr) const
 {
   if (vert2element)
-    return (*vert2element)[vnr-1];
+    return (*vert2element)[vnr];
   return FlatArray<int> (0,0);
 }
 
+FlatArray<int> MeshTopology :: GetVertexSurfaceElements (int vnr) const
+{
+  if (vert2surfelement)
+    return (*vert2surfelement)[vnr];
+  return FlatArray<int> (0,0);
+}
+
+
+void MeshTopology :: GetVertexSurfaceElements( int vnr, 
+					       ARRAY<int>& elements ) const
+{
+  if (vert2surfelement)
+    {
+      int i;
+      int ne = vert2surfelement->EntrySize(vnr);
+      elements.SetSize(ne);
+      for (i = 1; i <= ne; i++)
+	elements.Elem(i) = vert2surfelement->Get(vnr, i);
+    }
+}
 
 }
diff --git a/Netgen/libsrc/meshing/topology.hpp b/Netgen/libsrc/meshing/topology.hpp
index 751034be49..3c5ac1397f 100644
--- a/Netgen/libsrc/meshing/topology.hpp
+++ b/Netgen/libsrc/meshing/topology.hpp
@@ -28,12 +28,13 @@ class MeshTopology
   MoveableArray<int> surffaces;
   MoveableArray<INDEX_2> surf2volelement;
   MoveableArray<int> face2surfel;
-  TABLE<int> *vert2element;
-  TABLE<int> *vert2surfelement;
-  TABLE<int> *vert2segment;
+  TABLE<int,PointIndex::BASE> *vert2element;
+  TABLE<int,PointIndex::BASE> *vert2surfelement;
+  TABLE<int,PointIndex::BASE> *vert2segment;
   int timestamp;
 public:
   MeshTopology (const Mesh & amesh);
+  ~MeshTopology ();
 
   void SetBuildEdges (int be)
   { buildedges = be; }
@@ -104,6 +105,9 @@ public:
   
   void GetVertexElements (int vnr, ARRAY<int> & elements) const;
   FlatArray<int> GetVertexElements (int vnr) const;
+
+  void GetVertexSurfaceElements( int vnr, ARRAY<int>& elements ) const;
+  FlatArray<int> GetVertexSurfaceElements (int vnr) const;
 };
 
 #endif
diff --git a/Netgen/libsrc/meshing/zrefine.cpp b/Netgen/libsrc/meshing/zrefine.cpp
index 70dc91e1ca..7b4d4804a9 100644
--- a/Netgen/libsrc/meshing/zrefine.cpp
+++ b/Netgen/libsrc/meshing/zrefine.cpp
@@ -28,7 +28,7 @@ namespace netgen
     for (i = 1; i <= mesh.GetNSeg(); i++)
       {
 	const Segment & seg = mesh.LineSegment(i);
-	if (seg.singedge)
+	if (seg.singedge_left || seg.singedge_right)
 	  {
 	    INDEX_2 i2(seg.p1, seg.p2);
 	    i2.Sort();
diff --git a/Netgen/libsrc/occ/occgenmesh.cpp b/Netgen/libsrc/occ/occgenmesh.cpp
index f7055eb027..822edf3681 100644
--- a/Netgen/libsrc/occ/occgenmesh.cpp
+++ b/Netgen/libsrc/occ/occgenmesh.cpp
@@ -4,6 +4,9 @@
 #include <occgeom.hpp>
 #include <meshing.hpp>
 
+#include <stlgeom.hpp>
+
+
 namespace netgen
 {
 
@@ -12,15 +15,23 @@ namespace netgen
 #define TCL_OK 0
 #define TCL_ERROR 1
 
+extern STLParameters stlparam;
+
+#define DIVIDEEDGESECTIONS 1000
+#define IGNORECURVELENGTH 1e-4
 
-void DivideEdge (TopoDS_Edge & edge, ARRAY <MeshPoint> & ps, ARRAY<double> & params, Mesh & mesh)
+ 
+void DivideEdge (TopoDS_Edge & edge,
+		 ARRAY<MeshPoint> & ps,
+		 ARRAY<double> & params,
+		 Mesh & mesh)
 {
   double s0, s1;
   int j;
   double maxh = mparam.maxh;
   int nsubedges = 1;  
-  gp_Pnt pnt;
-  double svalue[1000];
+  gp_Pnt pnt, oldpnt;
+  double svalue[DIVIDEEDGESECTIONS];
 
   GProp_GProps system;
   BRepGProp::LinearProperties(edge, system);
@@ -28,49 +39,64 @@ void DivideEdge (TopoDS_Edge & edge, ARRAY <MeshPoint> & ps, ARRAY<double> & par
 
   Handle(Geom_Curve) c = BRep_Tool::Curve(edge, s0, s1);
 
-  double s = s0;
-  j = 0;
-  while (s < s1)
-    {
-      pnt = c->Value(s);
-      svalue[j] = s;
-      s += mesh.GetH(Point3d(pnt.X(), pnt.Y(), pnt.Z()))*(s1-s0)/L;
-      j++;
-      if (s < s1) nsubedges++;
-    }
-  svalue[j] = s;
-  double stretchfac = (s1-s0)/(s-s0);
+  double hvalue[DIVIDEEDGESECTIONS+1];
+  hvalue[0] = 0;
+  pnt = c->Value(s0);
 
-  if (nsubedges < mparam.segmentsperedge)
+  double olddist = 0;
+  double dist = 0;
+
+  for (int i = 1; i <= DIVIDEEDGESECTIONS; i++)
     {
-      nsubedges = 1;
-      s = s0;
-      j = 0;
-      while (s < s1)
-	{
-	  pnt = c->Value(s);
-	  svalue[j] = s;
-	  s += min ((s1-s0)/mparam.segmentsperedge, mesh.GetH(Point3d(pnt.X(), pnt.Y(), pnt.Z()))*(s1-s0)/L);
-	  j++;
-	  if (s < s1) nsubedges++; 
-	}
-      svalue[j] = s;
-      stretchfac = (s1-s0)/(s-s0);
+      oldpnt = pnt;
+      pnt = c->Value(s0+(i/double(DIVIDEEDGESECTIONS))*(s1-s0));
+      hvalue[i] = hvalue[i-1] +
+	1.0/mesh.GetH(Point3d(pnt.X(), pnt.Y(), pnt.Z()))*
+	pnt.Distance(oldpnt);
+
+      olddist = dist;
+      dist = pnt.Distance(oldpnt);
     }
 
-  ps.SetSize (nsubedges-1);
-  params.SetSize (nsubedges+1);
+  //  nsubedges = int(ceil(hvalue[DIVIDEEDGESECTIONS]));
+  nsubedges = max (1, int(floor(hvalue[DIVIDEEDGESECTIONS]+0.5)));
 
-  for (j = 1; j < nsubedges; j++)
+  ps.SetSize(nsubedges-1);
+  params.SetSize(nsubedges+1);
+
+  int i = 1;
+  int i1 = 0;
+  do
     {
-      params[j] = s0+stretchfac*(svalue[j]-s0);
-      pnt = c->Value(params[j]);
-      ps[j-1] = MeshPoint (Point3d(pnt.X(), pnt.Y(), pnt.Z()));
-    } 
+      if (hvalue[i1]/hvalue[DIVIDEEDGESECTIONS]*nsubedges >= i)
+	{
+	  params[i] = s0+(i1/double(DIVIDEEDGESECTIONS))*(s1-s0);
+	  pnt = c->Value(params[i]);
+	  ps[i-1] = MeshPoint (Point3d(pnt.X(), pnt.Y(), pnt.Z()));
+	  i++;
+	}
+      i1++;
+      if (i1 > DIVIDEEDGESECTIONS)
+	{
+	  nsubedges = i;
+	  ps.SetSize(nsubedges-1);
+	  params.SetSize(nsubedges+1);
+	  cout << "divide edge: local h too small" << endl;
+	}
+    } while (i < nsubedges);
 
   params[0] = s0;
   params[nsubedges] = s1;
+
+  if (params[nsubedges] <= params[nsubedges-1])
+    {
+      cout << "CORRECTED" << endl;
+      ps.SetSize (nsubedges-2);
+      params.SetSize (nsubedges);
+      params[nsubedges] = s1;
+    }
 }
+ 
 
 
 
@@ -78,6 +104,11 @@ static void FindEdges (OCCGeometry & geom, Mesh & mesh)
 {
   int i, j;
   
+  char * savetask = multithread.task;
+  multithread.task = "Edge meshing";
+
+  (*testout) << "edge meshing" << endl;
+
   TopExp_Explorer exp0, exp01, exp1, exp2, exp3;
   
   int nvertices = geom.vmap.Extent();
@@ -86,109 +117,245 @@ static void FindEdges (OCCGeometry & geom, Mesh & mesh)
     {
       gp_Pnt pnt = BRep_Tool::Pnt (TopoDS::Vertex(geom.vmap(i)));
       MeshPoint mp( Point3d(pnt.X(), pnt.Y(), pnt.Z()) );
+
+      /*
+      int exists = 0;
+			
+      for (j = 1; !exists && (j <= mesh.GetNP()); j++)
+	if ((mesh.Point(j)-Point<3>(mp)).Length() < 1e-6)
+	  exists = 1;
+      
+      if (!exists)
+      */
+
       mesh.AddPoint (mp);
     }
   
   int facenr = 0;
   int edgenr = 0;
   
+  int total = 0;
+  int solidnr = 0;
+
+  ARRAY<int> face2solid;
+  face2solid.SetSize (geom.fmap.Extent());
+  face2solid = 0;
+
+  /*
   for (exp0.Init(geom.shape, TopAbs_SOLID); exp0.More(); exp0.Next())
-    for (exp01.Init(exp0.Current(), TopAbs_SHELL); exp01.More(); exp01.Next())
-      {
-	TopoDS_Shape shell = exp01.Current();
-	
-	for (exp1.Init(shell, TopAbs_FACE); exp1.More(); exp1.Next())
+    {
+      solidnr++;
+      for (exp01.Init(exp0.Current(), TopAbs_SHELL); exp01.More(); exp01.Next())
+	{
+	  TopoDS_Shape shell = exp01.Current();
+	  for (exp1.Init(shell, TopAbs_FACE); exp1.More(); exp1.Next())
+	    {
+	      TopoDS_Face face = TopoDS::Face(exp1.Current());
+	      facenr = geom.fmap.FindIndex(face);
+	      face2solid[facenr-1] = solidnr;
+	    
+	      for (exp2.Init (face, TopAbs_WIRE); exp2.More(); exp2.Next())
+		{
+		  TopoDS_Shape wire = exp2.Current();
+		  
+		  for (exp3.Init (wire, TopAbs_EDGE); exp3.More(); exp3.Next())
+		    {
+		      total++;
+		    }
+		}
+	    }
+	  }
+      }
+  */
+
+  for (exp0.Init(geom.shape, TopAbs_SOLID); exp0.More(); exp0.Next())
+    {
+      solidnr++;
+      for (exp1.Init(exp0.Current(), TopAbs_FACE); exp1.More(); exp1.Next())
+	{
+	  TopoDS_Face face = TopoDS::Face(exp1.Current());
+	  facenr = geom.fmap.FindIndex(face);
+	  face2solid[facenr-1] = solidnr;
+	}
+      }
+
+
+  for (int i3 = 1; i3 <= geom.fmap.Extent(); i3++)
+    for (exp2.Init (geom.fmap(i3), TopAbs_WIRE); exp2.More(); exp2.Next())
+      for (exp3.Init (exp2.Current(), TopAbs_EDGE); exp3.More(); exp3.Next())
+	total++;
+	   
+
+  int curr = 0;
+
+  
+  solidnr = 0;
+  /*
+  for (exp0.Init(geom.shape, TopAbs_SOLID); exp0.More(); exp0.Next())
+    {
+      solidnr++;
+      for (exp01.Init(exp0.Current(), TopAbs_SHELL); exp01.More(); exp01.Next())
+	{
+	  TopoDS_Shape shell = exp01.Current();
+	  
+	  for (exp1.Init(shell, TopAbs_FACE); exp1.More(); exp1.Next())
+	    {
+	      
+	      TopoDS_Face face = TopoDS::Face(exp1.Current());
+  */
+	      
+  for (int i3 = 1; i3 <= geom.fmap.Extent(); i3++)
+    
+      
+	{
 	  {
-	    TopoDS_Face face = TopoDS::Face(exp1.Current());
+	    {
+	      TopoDS_Face face = TopoDS::Face(geom.fmap(i3));
+	      solidnr = face2solid[i3-1];
+	      
+	      
+	      facenr = geom.fmap.FindIndex (face);
+	      
+	      mesh.AddFaceDescriptor (FaceDescriptor(facenr, solidnr, 0, 0));
+	      Handle(Geom_Surface) occface = BRep_Tool::Surface(face);
+	      
+	      for (exp2.Init (face, TopAbs_WIRE); exp2.More(); exp2.Next())
+		{
+		  TopoDS_Shape wire = exp2.Current();
+		  
+		  for (exp3.Init (wire, TopAbs_EDGE); exp3.More(); exp3.Next())
+		    {
+		      curr++;
+ 
+		      multithread.percent = 100 * curr / (double) total;
+		      if (multithread.terminate) return;
 
-	    facenr = geom.fmap.FindIndex (face);
-	    
-	    mesh.AddFaceDescriptor (FaceDescriptor(facenr, 1, 0, 0));
-	    Handle(Geom_Surface) occface = BRep_Tool::Surface(face);
-	    
-	    for (exp2.Init (face, TopAbs_WIRE); exp2.More(); exp2.Next())
-	      {
-		TopoDS_Shape wire = exp2.Current();
+		      TopoDS_Edge edge = TopoDS::Edge (exp3.Current());
+		      if (BRep_Tool::Degenerated(edge)) continue;
 
-		for (exp3.Init (wire, TopAbs_EDGE); exp3.More(); exp3.Next())
-		  {
-		    TopoDS_Edge edge = TopoDS::Edge (exp3.Current());
-		    if (BRep_Tool::Degenerated(edge)) continue;
+		      if (geom.vmap.FindIndex(TopExp::FirstVertex (edge)) == 
+			  geom.vmap.FindIndex(TopExp::LastVertex (edge)))
+			{
+			  GProp_GProps system;
+			  BRepGProp::LinearProperties(edge, system);
+
+			  if (system.Mass() < 1e-5)
+			    {
+			      cout << "ignoring edge " << geom.emap.FindIndex (edge)
+				   << ". closed edge with length < 1e-5" << endl;
+			      continue;
+			    }
+			}
 
-		    Handle(Geom2d_Curve) cof;
-		    double s0, s1;
-		    cof = BRep_Tool::CurveOnSurface (edge, face, s0, s1);
-		    
-		    int geomedgenr = geom.emap.FindIndex(edge);
 
-		    ARRAY <MeshPoint> mp;
-		    ARRAY <double> params;
-		    
-		    DivideEdge (edge, mp, params, mesh);
+		      Handle(Geom2d_Curve) cof;
+		      double s0, s1;
+		      cof = BRep_Tool::CurveOnSurface (edge, face, s0, s1);
+
+		      int geomedgenr = geom.emap.FindIndex(edge);
+
+		      ARRAY <MeshPoint> mp;
+		      ARRAY <double> params;
+
+		      DivideEdge (edge, mp, params, mesh);
+
 		    
-		    ARRAY <int> pnums;
-		    pnums.SetSize (mp.Size()+2);
+		      ARRAY <int> pnums;
+		      pnums.SetSize (mp.Size()+2);
 		    
-		    pnums[0] = geom.vmap.FindIndex (TopExp::FirstVertex (edge));
-		    pnums[pnums.Size()-1] = geom.vmap.FindIndex (TopExp::LastVertex (edge));
+		      pnums[0] = geom.vmap.FindIndex (TopExp::FirstVertex (edge));
+		      pnums[pnums.Size()-1] = geom.vmap.FindIndex (TopExp::LastVertex (edge));
 
-		    for (i = 1; i <= mp.Size(); i++)
-		      {
-			int exists = 0;
+		      for (i = 1; i <= mp.Size(); i++)
+			{
+			  int exists = 0;
 			
-			for (j = 1; !exists && (j <= mesh.GetNP()-nvertices); j++)
-			  if ((mesh.Point(nvertices+j)-Point<3>(mp[i-1])).Length() < 1e-6) exists = 1;
-			if (exists)
-			  {
-			    pnums[i] = nvertices+j-1;
-			  }
-			else
-			  {
-			    mesh.AddPoint (mp[i-1]);
-			    pnums[i] = mesh.GetNP();
-			  }
-		      }
+			  for (j = 1; !exists && (j <= mesh.GetNP()-nvertices); j++)
+			    if ((mesh.Point(nvertices+j)-Point<3>(mp[i-1])).Length() < 1e-6) exists = 1;
+
+			  if (exists)
+			    {
+			      pnums[i] = nvertices+j-1;
+			    }
+			  else
+			    {
+			      mesh.AddPoint (mp[i-1]);
+			      pnums[i] = mesh.GetNP();
+			    }
+			}
 		    
-		    for (i = 1; i <= mp.Size()+1; i++)
-		      {
-			edgenr++;
-			Segment seg;
-			
-			seg.p1 = pnums[i-1];
-			seg.p2 = pnums[i];
-			seg.edgenr = edgenr;
-			seg.si = facenr;
-			seg.epgeominfo[0].dist = params[i-1];
-			seg.epgeominfo[1].dist = params[i];
-			seg.epgeominfo[0].edgenr = geomedgenr;
-			seg.epgeominfo[1].edgenr = geomedgenr;
-			
-			gp_Pnt2d p2d;
-			p2d = cof->Value(params[i-1]);
-			seg.epgeominfo[0].u = p2d.X();
-			seg.epgeominfo[0].v = p2d.Y();
-			p2d = cof->Value(params[i]);
-			seg.epgeominfo[1].u = p2d.X();
-			seg.epgeominfo[1].v = p2d.Y();
-			
-			if (edge.Orientation() == TopAbs_REVERSED)
-			  {
-			    swap (seg.p1, seg.p2);
-			    swap (seg.epgeominfo[0].dist, seg.epgeominfo[1].dist);
-			    swap (seg.epgeominfo[0].u, seg.epgeominfo[1].u);
-			    swap (seg.epgeominfo[0].v, seg.epgeominfo[1].v);
-			  }
+		      for (i = 1; i <= mp.Size()+1; i++)
+			{
+			  edgenr++;
+			  Segment seg;
 			
+			  seg.p1 = pnums[i-1];
+			  seg.p2 = pnums[i];
+			  seg.edgenr = edgenr;
+			  seg.si = facenr;
+			  seg.epgeominfo[0].dist = params[i-1];
+			  seg.epgeominfo[1].dist = params[i];
+			  seg.epgeominfo[0].edgenr = geomedgenr;
+			  seg.epgeominfo[1].edgenr = geomedgenr;
 			
-			mesh.AddSegment (seg);
+			  gp_Pnt2d p2d;
+			  p2d = cof->Value(params[i-1]);
+			  //			if (i == 1) p2d = cof->Value(s0);
+			  seg.epgeominfo[0].u = p2d.X();
+			  seg.epgeominfo[0].v = p2d.Y();
+			  p2d = cof->Value(params[i]);
+			  //			if (i == mp.Size()+1) p2d = cof -> Value(s1);
+			  seg.epgeominfo[1].u = p2d.X();
+			  seg.epgeominfo[1].v = p2d.Y();
+
+			  /*
+			    if (occface->IsUPeriodic())
+			    {
+			    cout << "U Periodic" << endl;
+			    if (fabs(seg.epgeominfo[1].u-seg.epgeominfo[0].u) >
+			    fabs(seg.epgeominfo[1].u-
+			    (seg.epgeominfo[0].u-occface->UPeriod())))
+			    seg.epgeominfo[0].u = p2d.X()+occface->UPeriod();
+
+			    if (fabs(seg.epgeominfo[1].u-seg.epgeominfo[0].u) >
+			    fabs(seg.epgeominfo[1].u-
+			    (seg.epgeominfo[0].u+occface->UPeriod())))
+			    seg.epgeominfo[0].u = p2d.X()-occface->UPeriod();
+			    }
+
+			    if (occface->IsVPeriodic())
+			    {
+			    cout << "V Periodic" << endl;
+			    if (fabs(seg.epgeominfo[1].v-seg.epgeominfo[0].v) >
+			    fabs(seg.epgeominfo[1].v-
+			    (seg.epgeominfo[0].v-occface->VPeriod())))
+			    seg.epgeominfo[0].v = p2d.Y()+occface->VPeriod();
+
+			    if (fabs(seg.epgeominfo[1].v-seg.epgeominfo[0].v) >
+			    fabs(seg.epgeominfo[1].v-
+			    (seg.epgeominfo[0].v+occface->VPeriod())))
+			    seg.epgeominfo[0].v = p2d.Y()-occface->VPeriod();
+			    }
+			  */
 			
-			// (*testout) << "seg" << seg.edgenr << ": " << seg.p1 << " - " << seg.p2 << endl;
-		      }
-		  }
-	      }
-	  }
-      }
+			  if (edge.Orientation() == TopAbs_REVERSED)
+			    {
+			      swap (seg.p1, seg.p2);
+			      swap (seg.epgeominfo[0].dist, seg.epgeominfo[1].dist);
+			      swap (seg.epgeominfo[0].u, seg.epgeominfo[1].u);
+			      swap (seg.epgeominfo[0].v, seg.epgeominfo[1].v);
+			    }
+
+			  mesh.AddSegment (seg);
+
+			}
+		    }
+		}
+	    }
+	}
+    }
   mesh.CalcSurfacesOfNode();
+  multithread.task = savetask;
 }  
 
 
@@ -202,94 +369,238 @@ static void OCCMeshSurface (OCCGeometry & geom, Mesh & mesh, int perfstepsend)
   char * savetask = multithread.task;
   multithread.task = "Surface meshing";
   
+  geom.facemeshstatus = 0;
+
   int noldp = mesh.GetNP();
 
   double starttime = GetTime();
 
   ARRAY<int> glob2loc(noldp);
 
+  int projecttype = PARAMETERSPACE;
+  
+  //int projecttype = PLANESPACE;
+
+  int notrys = 1;
+
+  int surfmesherror = 0;
+
   for (k = 1; k <= mesh.GetNFD(); k++)
     {
-
       (*testout) << "mesh face " << k << endl;
       multithread.percent = 100 * k / (mesh.GetNFD()+1e-10);
+      geom.facemeshstatus[k-1] = -1;
 
+      
+      /*
+      if (k != 138)
+	{
+	  cout << "skipped" << endl;
+	  continue;
+	}
+      */
+    
       FaceDescriptor & fd = mesh.GetFaceDescriptor(k);
 
-      PrintMessage (2, "Surface ", k, " / ", mesh.GetNFD());
-
       int oldnf = mesh.GetNSE();
 
-      
-      Box<3> bb (Point<3> (-geom.MaxSize(), -geom.MaxSize(), -geom.MaxSize()), 
-		 Point<3> (geom.MaxSize(), geom.MaxSize(), geom.MaxSize()));
+      Box<3> bb = geom.GetBoundingBox();
+ 
+      Meshing2OCCSurfaces meshing(TopoDS::Face(geom.fmap(k)), bb, projecttype); 
+
+      if (meshing.GetProjectionType() == PLANESPACE)
+	PrintMessage (2, "Face ", k, " / ", mesh.GetNFD(), " (plane space projection)");
+      else
+	PrintMessage (2, "Face ", k, " / ", mesh.GetNFD(), " (parameter space projection)");
+
+      if (surfmesherror)
+	cout << "Surface meshing error occured before (in " << surfmesherror << " faces)" << endl;
 
-      Meshing2OCCSurfaces meshing(TopoDS::Face(geom.fmap(k)), bb); 
       //      Meshing2OCCSurfaces meshing(f2, bb); 
       meshing.SetStartTime (starttime);
 
-      /*
-      for (i = 1; i <= noldp; i++)
-	{
-	  // (*testout) << "Add point " << mesh.Point(i) << endl;
-	  meshing.AddPoint (mesh.Point(i), i);
-	}
-      */
+      (*testout) << "Face " << k << endl << endl;
+
 
-      int cntp = 0;
-      glob2loc = 0;
-      for (i = 1; i <= mesh.GetNSeg(); i++)
+      if (meshing.GetProjectionType() == PLANESPACE)
 	{
-	  Segment & seg = mesh.LineSegment(i);
-	  if (seg.si == k)
+	  int cntp = 0;
+	  glob2loc = 0;
+	  for (i = 1; i <= mesh.GetNSeg(); i++)
 	    {
-	      for (j = 1; j <= 2; j++)
+	      Segment & seg = mesh.LineSegment(i);
+	      if (seg.si == k)
 		{
-		  int pi = (j == 1) ? seg.p1 : seg.p2;
-		  if (!glob2loc.Get(pi))
+		  for (j = 1; j <= 2; j++)
 		    {
-		      meshing.AddPoint (mesh.Point(pi), pi);
-		      cntp++;
-		      glob2loc.Elem(pi) = cntp;
+		      int pi = (j == 1) ? seg.p1 : seg.p2;
+		      if (!glob2loc.Get(pi))
+			{
+			  meshing.AddPoint (mesh.Point(pi), pi);
+			  cntp++;
+			  glob2loc.Elem(pi) = cntp;
+			}
 		    }
 		}
 	    }
+	  
+	  for (i = 1; i <= mesh.GetNSeg(); i++)
+	    {
+	      Segment & seg = mesh.LineSegment(i);
+	      if (seg.si == k)
+		{
+		  PointGeomInfo gi0, gi1;
+		  gi0.trignum = gi1.trignum = k;
+		  gi0.u = seg.epgeominfo[0].u;
+		  gi0.v = seg.epgeominfo[0].v;
+		  gi1.u = seg.epgeominfo[1].u;
+		  gi1.v = seg.epgeominfo[1].v;
+		  
+		  meshing.AddBoundaryElement (glob2loc.Get(seg.p1), glob2loc.Get(seg.p2), gi0, gi1);
+		  (*testout) << gi0.u << " " << gi0.v << endl;
+		  (*testout) << gi1.u << " " << gi1.v << endl;
+		}
+	    }
 	}
-
-      for (i = 1; i <= mesh.GetNSeg(); i++)
+      else
 	{
-	  Segment & seg = mesh.LineSegment(i);
-	  if (seg.si == k)
+	  int cntp = 0;
+	  
+	  for (i = 1; i <= mesh.GetNSeg(); i++)
+	    if (mesh.LineSegment(i).si == k)
+	      cntp+=2;
+	  
+	  
+	  ARRAY< PointGeomInfo > gis;
+	  
+	  gis.SetAllocSize (cntp);
+	  gis.SetSize (0);
+	  
+	  for (i = 1; i <= mesh.GetNSeg(); i++)
 	    {
-	      PointGeomInfo gi0, gi1;
-	      gi0.trignum = gi1.trignum = k;
-	      gi0.u = seg.epgeominfo[0].u;
-	      gi0.v = seg.epgeominfo[0].v;
-	      gi1.u = seg.epgeominfo[1].u;
-	      gi1.v = seg.epgeominfo[1].v;
-
-	      meshing.AddBoundaryElement (glob2loc.Get(seg.p1), glob2loc.Get(seg.p2), gi0, gi1);
+	      Segment & seg = mesh.LineSegment(i);
+	      if (seg.si == k)
+		{
+		  PointGeomInfo gi0, gi1;
+		  gi0.trignum = gi1.trignum = k;
+		  gi0.u = seg.epgeominfo[0].u;
+		  gi0.v = seg.epgeominfo[0].v;
+		  gi1.u = seg.epgeominfo[1].u;
+		  gi1.v = seg.epgeominfo[1].v;
+		  
+		  int locpnum[2] = {0, 0};
+		  
+		  for (j = 0; j < 2; j++)
+		    {
+		      PointGeomInfo gi = (j == 0) ? gi0 : gi1;
+		      
+		      int l;
+		      for (l = 0; l < gis.Size() && locpnum[j] == 0; l++)
+			{
+			  double dist = sqr (gis[l].u-gi.u)+sqr(gis[l].v-gi.v);
+
+			  if (dist < 1e-10)
+			    locpnum[j] = l+1;
+			}
+		      
+		      if (locpnum[j] == 0)
+			{
+			  int pi = (j == 0) ? seg.p1 : seg.p2;
+			  meshing.AddPoint (mesh.Point(pi), pi);
+
+			  gis.SetSize (gis.Size()+1);
+			  gis[l] = gi;
+			  locpnum[j] = l+1;
+			}
+		    }
+		  
+		  meshing.AddBoundaryElement (locpnum[0], locpnum[1], gi0, gi1);
+		  (*testout) << gi0.u << " " << gi0.v << endl;
+		  (*testout) << gi1.u << " " << gi1.v << endl;
 	      
-	      (*testout) << "Add BE " << seg.p1 << "-" << seg.p2 << " (" << ")" << endl;
+		}
 	    }
 	}
 
+
+
+
+
+
       double maxh = mparam.maxh;
+      mparam.checkoverlap = 0;
+      //      int noldpoints = mesh->GetNP();
+      int noldsurfel = mesh.GetNSE();
 
-      MESHING2_RESULT res =
-	meshing.GenerateMesh (mesh, maxh, k);
+      MESHING2_RESULT res;
+
+      try {
+        res = meshing.GenerateMesh (mesh, maxh, k);
+      }
+
+      catch (SingularMatrixException)
+	{
+	  (*myerr) << "Singular Matrix" << endl;
+	  res = MESHING2_GIVEUP;
+	}
+
+      catch (UVBoundsException)
+	{
+	  (*myerr) << "UV bounds exceeded" << endl;
+	  res = MESHING2_GIVEUP;
+	}
+
+      projecttype = PARAMETERSPACE;
 
       if (res != MESHING2_OK)
 	{
-	  PrintError ("Problem in Surface mesh generation");
-	  throw NgException ("Problem in Surface mesh generation");
+	  if (notrys == 1)
+	    {
+	      for (int i = noldsurfel+1; i <= mesh.GetNSE(); i++)
+		mesh.DeleteSurfaceElement (i);
+	      
+	      mesh.Compress();
+
+	      cout << "retry Surface " << k << endl;
+
+	      k--;
+	      projecttype*=-1;
+	      notrys++;
+	      continue;
+	    }
+	  else
+	    {
+	      geom.facemeshstatus[k-1] = -1;
+	      PrintError ("Problem in Surface mesh generation");
+	      surfmesherror++;
+	      //	      throw NgException ("Problem in Surface mesh generation");
+	    }
 	}
+      else
+	{
+	  geom.facemeshstatus[k-1] = 1;
+ 	}
+
+      notrys = 1;
       
       for (i = oldnf+1; i <= mesh.GetNSE(); i++)
 	mesh.SurfaceElement(i).SetIndex (k);
 
     }
 
+  if (surfmesherror)
+    {
+      cout << "WARNING! NOT ALL FACES HAVED BEEN MESHED" << endl;
+      cout << "SURFACE MESHING ERROR OCCURED IN " << surfmesherror << " FACES:" << endl;
+      for (int i = 1; i <= geom.fmap.Extent(); i++)
+	  if (geom.facemeshstatus[i-1] == -1)
+	    cout << "Face " << i << endl;
+      cout << endl;
+      cout << "for more information open IGES/STEP Topology Explorer" << endl;
+      throw NgException ("Problem in Surface mesh generation");
+    }
+
+
   if (multithread.terminate || perfstepsend < MESHCONST_OPTSURFACE)
     return;
 
@@ -371,11 +682,126 @@ double ComputeH (double kappa)
   
   if (mparam.maxh < hret)
     hret = mparam.maxh;
-
+  
   return (hret);
 }
 
 
+class Line
+{
+public:
+  Point<3> p0, p1;
+  double Dist (Line l)
+  {
+    Vec<3> n = p1-p0;
+    Vec<3> q = l.p1-l.p0;
+    double nq = n*q;
+    
+    Point<3> p = p0 + 0.5*n;
+    double lambda = (p-l.p0)*n / nq;
+
+    if (lambda >= 0 && lambda <= 1)
+      {
+	double d = (p-l.p0-lambda*q).Length();
+	//        if (d < 1e-3) d = 1e99;
+	return d;
+      }
+    else
+      return 1e99;
+  }
+  
+  double Length ()
+  {
+    return (p1-p0).Length();
+  };
+};
+
+
+
+void RestrictHTriangle (gp_Pnt2d & par0, gp_Pnt2d & par1, gp_Pnt2d & par2,
+			BRepLProp_SLProps * prop, Mesh & mesh, double maxside, int depth, double h = 0)
+{
+  gp_Pnt2d parmid;
+
+  parmid.SetX(0.3*(par0.X()+par1.X()+par2.X()));
+  parmid.SetY(0.3*(par0.Y()+par1.Y()+par2.Y()));
+  
+  if (depth == 0)
+    {
+      prop->SetParameters (parmid.X(), parmid.Y());
+      if (!prop->IsCurvatureDefined())
+	{
+	  (*testout) << "curvature not defined!" << endl;
+	  return;
+	}
+      double curvature = max(fabs(prop->MinCurvature()), fabs(prop->MaxCurvature()));
+      if (curvature < 1e-3)
+	{
+	  (*testout) << "curvature too small (" << curvature << ")!" << endl;
+	  //	  return;
+	}
+      h = ComputeH (curvature+1e-10);
+    }
+  
+  if (h < maxside)
+    {
+      gp_Pnt2d pm0;
+      gp_Pnt2d pm1;
+      gp_Pnt2d pm2;
+      
+      pm0.SetX(0.5*(par1.X()+par2.X())); pm0.SetY(0.5*(par1.Y()+par2.Y()));
+      pm1.SetX(0.5*(par0.X()+par2.X())); pm1.SetY(0.5*(par0.Y()+par2.Y()));
+      pm2.SetX(0.5*(par1.X()+par0.X())); pm2.SetY(0.5*(par1.Y()+par0.Y()));
+
+      RestrictHTriangle (pm0, pm1, pm2, prop, mesh, maxside/2, depth+1, h);
+      RestrictHTriangle (par0, pm1, pm2, prop, mesh, maxside/2, depth+1, h);
+      RestrictHTriangle (par1, pm0, pm2, prop, mesh, maxside/2, depth+1, h);
+      RestrictHTriangle (par2, pm1, pm0, prop, mesh, maxside/2, depth+1, h);
+    }
+  else
+    {
+      prop->SetParameters (parmid.X(), parmid.Y());
+      gp_Pnt pnt = prop->Value();
+      Point3d p3d(pnt.X(), pnt.Y(), pnt.Z());
+      mesh.RestrictLocalH (p3d, h);
+
+      prop->SetParameters (par0.X(), par0.Y());
+      pnt = prop->Value();
+      p3d = Point3d(pnt.X(), pnt.Y(), pnt.Z());
+      mesh.RestrictLocalH (p3d, h);
+
+      prop->SetParameters (par1.X(), par1.Y());
+      pnt = prop->Value();
+      p3d = Point3d(pnt.X(), pnt.Y(), pnt.Z());
+      mesh.RestrictLocalH (p3d, h);
+
+      prop->SetParameters (par2.X(), par2.Y());
+      pnt = prop->Value();
+      p3d = Point3d(pnt.X(), pnt.Y(), pnt.Z());
+      mesh.RestrictLocalH (p3d, h);
+
+      //      (*testout) << "p = " << p3d << ", h = " << h << ", maxside = " << maxside << endl;
+      /*
+      (*testout) << pnt.X() << " " << pnt.Y() << " " << pnt.Z() << endl;
+
+      prop->SetParameters (par0.X(), par0.Y());
+      pnt = prop->Value();
+      (*testout) << pnt.X() << " " << pnt.Y() << " " << pnt.Z() << endl;
+
+      prop->SetParameters (par1.X(), par1.Y());
+      pnt = prop->Value();
+      (*testout) << pnt.X() << " " << pnt.Y() << " " << pnt.Z() << endl;
+
+      prop->SetParameters (par2.X(), par2.Y());
+      pnt = prop->Value();
+      (*testout) << pnt.X() << " " << pnt.Y() << " " << pnt.Z() << endl;
+      */
+    }
+}
+
+
+
+
 int OCCGenerateMesh (OCCGeometry & geom,
 		  Mesh *& mesh,
 		  int perfstepsstart, int perfstepsend,
@@ -383,6 +809,8 @@ int OCCGenerateMesh (OCCGeometry & geom,
 {
   int i, j;
 
+  multithread.percent = 0;
+
   if (perfstepsstart <= MESHCONST_ANALYSE)
     {
       delete mesh;
@@ -390,42 +818,128 @@ int OCCGenerateMesh (OCCGeometry & geom,
 
       mesh->SetGlobalH (mparam.maxh);
 
-      ARRAY<double> maxhdom(1);
-      maxhdom[0] = mparam.maxh;
-      double maxsize = geom.MaxSize(); 
+      ARRAY<double> maxhdom;
+      maxhdom.SetSize (geom.NrSolids());
+      maxhdom = mparam.maxh;
       
       mesh->SetMaxHDomain (maxhdom);
-      mesh->SetLocalH (Point<3>(-maxsize, -maxsize, -maxsize),
-		       Point<3>(maxsize, maxsize, maxsize),
-		       0.5);
+      
+      Box<3> bb = geom.GetBoundingBox();
+      bb.Increase (bb.Diam()/10);
+      
+      mesh->SetLocalH (bb.PMin(), bb.PMax(), 0.5);
+    
 
       if (mparam.uselocalh)
 	{
 
 	  char * savetask = multithread.task;
-	  multithread.task = "Setting local mesh size";
-  
-	  double maxsize = geom.MaxSize(); 
-	  mesh->SetLocalH (Point<3>(-maxsize, -maxsize, -maxsize),
-			   Point<3>(maxsize, maxsize, maxsize),
-			   mparam.grading);
+	  multithread.percent = 0;
+
+	  mesh->SetLocalH (bb.PMin(), bb.PMax(), mparam.grading);
+
+	  int nedges = geom.emap.Extent();
+	  int i;
+
+	  double maxedgelen = 0;
+	  double minedgelen = 1e99;
+
+	  
+	  multithread.task = "Setting local mesh size (elements per edge)";
+
+	  // setting elements per edge
+
+	  for (i = 1; i <= nedges && !multithread.terminate; i++)
+	    {
+	      TopoDS_Edge e = TopoDS::Edge (geom.emap(i));
+	      multithread.percent = 100 * (i-1)/double(nedges);
+	      if (BRep_Tool::Degenerated(e)) continue;
+
+	      GProp_GProps system;	       
+	      BRepGProp::LinearProperties(e, system);
+	      double len = system.Mass();
+
+	      if (len < IGNORECURVELENGTH)
+		{
+		  (*testout) << "ignored" << endl;
+		  continue;
+		}
+
+
+	      double localh = len/mparam.segmentsperedge;
+	      double s0, s1;
+	      Handle(Geom_Curve) c = BRep_Tool::Curve(e, s0, s1);
+
+	      maxedgelen = max (maxedgelen, len);
+	      minedgelen = min (minedgelen, len);
+
+	      int j;
+	      int maxj = (int) ceil (localh);
+	      for (j = 0; j <= localh; j++)
+		{
+		  gp_Pnt pnt = c->Value (s0+double(j)/maxj*(s1-s0));
+		  mesh->RestrictLocalH (Point3d(pnt.X(), pnt.Y(), pnt.Z()), localh);
+		}
+	    }
+
+
+
+
+	  multithread.task = "Setting local mesh size (edge curvature)";
+
+	  
+	  // setting edge curvature
+
+	  //	  int nsections = 10;
+	  int nsections = 20;
+
+	  for (i = 1; i <= nedges && !multithread.terminate; i++)
+	    {
+	      multithread.percent = 100 * (i-1)/double(nedges);
+	      TopoDS_Edge edge = TopoDS::Edge (geom.emap(i));
+	      if (BRep_Tool::Degenerated(edge)) continue;
+	      double s0, s1;
+	      Handle(Geom_Curve) c = BRep_Tool::Curve(edge, s0, s1);
+	      BRepAdaptor_Curve brepc(edge);
+ 	      BRepLProp_CLProps prop(brepc, 2, 1e-5);
+	      
+	      int j;
+	      for (j = 1; j <= nsections; j++)
+		//	      for (j = 0; j < nsections; j++)
+		{
+		  double s = s0 + j/(double) nsections * (s1-s0);
+		  prop.SetParameter (s);
+		  double curvature = prop.Curvature();
+
+		  if (curvature >= 1e99) continue;
+
+		  gp_Pnt pnt = c->Value (s);
+
+		  mesh->RestrictLocalH (Point3d(pnt.X(), pnt.Y(), pnt.Z()),
+					ComputeH (fabs(curvature)));		  
+		}
+	    }
 
-	  cout << "maxsize = " << maxsize << endl;
+	  
+
+	  multithread.task = "Setting local mesh size (face curvature)";
+
+	  // setting face curvature
 	  
 	  int nfaces = geom.fmap.Extent();
 	  
-	  int i;
-	  for (i = 1; i <= nfaces; i++)
+	  for (i = 1; i <= nfaces && !multithread.terminate; i++)
 	    {
-	      (*testout) << "face " << i << endl;
 	      multithread.percent = 100 * (i-1)/double(nfaces);
 	      TopoDS_Face face = TopoDS::Face(geom.fmap(i));
 	      TopLoc_Location loc;
 	      Handle(Geom_Surface) surf = BRep_Tool::Surface (face);
 	      Handle(Poly_Triangulation) triangulation = BRep_Tool::Triangulation (face, loc);
-	      BRepAdaptor_Surface sf(face, Standard_False);
-	      BRepLProp_SLProps prop(sf, 2, 1e-5);
+	      if (triangulation.IsNull()) continue;
 	      
+	      BRepAdaptor_Surface sf(face, Standard_True);
+	      BRepLProp_SLProps prop(sf, 2, 1e-5);
+
 	      int ntriangles = triangulation -> NbTriangles();
 	      for (j = 1; j <= ntriangles; j++)
 		{
@@ -436,118 +950,122 @@ int OCCGenerateMesh (OCCGeometry & geom,
 		  for (k = 1; k <=3; k++)
 		    {
 		      int n = triangulation->Triangles()(j)(k);
-		      p[k-1] = triangulation->Nodes()(n);
+		      p[k-1] = triangulation->Nodes()(n).Transformed(loc);
 		      par[k-1] = triangulation->UVNodes()(n);
 		    }
 
-		  double lp2p0 = p[2].Distance(p[0]);
-		  double lp2p1 = p[2].Distance(p[1]);
-
-		  prop.SetParameters (p[2].X(), p[2].Y());
-		  double h;
-
-		  if (!prop.IsCurvatureDefined()) continue;
+		  double maxside = 0;
+		  maxside = max (maxside, p[0].Distance(p[1]));
+		  maxside = max (maxside, p[0].Distance(p[2]));
+		  maxside = max (maxside, p[1].Distance(p[2]));
 
-		  h = ComputeH (max(fabs(prop.MinCurvature()),
-				    fabs(prop.MaxCurvature()))+1e-10);
-		  
-		  double h0 = h/lp2p0;
-		  double h1 = h/lp2p1;
+		  RestrictHTriangle (par[0], par[1], par[2], &prop, *mesh, maxside, 0);
+		}
+	    }
 
-		  h0 = min (h0, 1.0);
-		  h1 = min (h1, 1.0);
 
-		  double l0, l1, l2;
-		  for (l0 = 0; l0 <= 1; l0+=h0)
-		    for (l1 = 0; l1 <= 1-l0; l1+=h1)
-		      {
-			l2 = 1-l0-l1;
-			double u = l0*par[0].X() + l1*par[1].X() + l2*par[2].X();
-			double v = l0*par[0].Y() + l1*par[1].Y() + l2*par[2].Y();
 
-			prop.SetParameters (u, v);
+	  // setting close edges
 
-			if (!prop.IsCurvatureDefined()) continue;
+	  if (stlparam.resthcloseedgeenable)
+	    {
+	      multithread.task = "Setting local mesh size (close edges)";
+	      
+	      int sections = 100;
+	      
+	      ARRAY<Line> lines(sections*nedges);
+	      
+	      Box3dTree* searchtree =
+		new Box3dTree (bb.PMin(), bb.PMax());
+	      
+	      int nlines = 0;
+	      for (int i = 1; i <= nedges && !multithread.terminate; i++)
+		{
+		  TopoDS_Edge edge = TopoDS::Edge (geom.emap(i));
+		  if (BRep_Tool::Degenerated(edge)) continue;
+
+		  double s0, s1;
+		  Handle(Geom_Curve) c = BRep_Tool::Curve(edge, s0, s1);
+		  BRepAdaptor_Curve brepc(edge);
+		  BRepLProp_CLProps prop(brepc, 1, 1e-5);
+		  prop.SetParameter (s0);
+
+		  gp_Vec d0 = prop.D1().Normalized();
+		  double s_start = s0;
+		  int count = 0;
+		  for (int j = 1; j <= sections; j++)
+		    {
+		      double s = s0 + (s1-s0)*(double)j/(double)sections;
+		      prop.SetParameter (s);
+		      gp_Vec d1 = prop.D1().Normalized();
+		      double cosalpha = fabs(d0*d1);
+		      if ((j == sections) || (cosalpha < cos(10.0/180.0*M_PI)))
+			{
+			  count++;
+			  gp_Pnt p0 = c->Value (s_start);
+			  gp_Pnt p1 = c->Value (s);
+			  lines[nlines].p0 = Point<3> (p0.X(), p0.Y(), p0.Z());
+			  lines[nlines].p1 = Point<3> (p1.X(), p1.Y(), p1.Z());
+		      
+			  Box3d box;
+			  box.SetPoint (Point3d(lines[nlines].p0));
+			  box.AddPoint (Point3d(lines[nlines].p1));
 
-			gp_Pnt pnt = prop.Value();
+			  searchtree->Insert (box.PMin(), box.PMax(), nlines+1);
+			  nlines++;
 
-			mesh->RestrictLocalH (Point3d(pnt.X(), pnt.Y(), pnt.Z()),
-					      ComputeH (max(fabs(prop.MinCurvature()),
-							    fabs(prop.MaxCurvature()))+1e-10));
-		      }
-		  
+			  s_start = s;
+			  d0 = d1;
+			}
+		    }
 		}
-
-	      /*
-	      // schmale lange drei+viereckerl
-	      TopExp_Explorer exp;
-	      for (exp.Init (face, TopAbs_WIRE); exp.More(); exp.Next())
+		
+	      ARRAY<int> linenums;
+	      
+	      for (int i = 0; i < nlines; i++)
 		{
-		  bool done = 0;
-		  BRepTools_WireExplorer wexp;
-		  double L[4];
-		  GProp_GProps system;
-		  int i = 0;
-		  double minL = 1e10;
-
-		  for (wexp.Init (TopoDS::Wire(exp.Current())); wexp.More() && i < 4; wexp.Next())
+		  multithread.percent = (100*i)/double(nlines);
+		  Line & line = lines[i];
+		  
+		  Box3d box;
+		  box.SetPoint (Point3d(line.p0));
+		  box.AddPoint (Point3d(line.p1));
+		  double maxhline = max (mesh->GetH(box.PMin()),
+					 mesh->GetH(box.PMax()));
+		  box.Increase(maxhline);
+
+		  double mindist = 1e99;
+		  linenums.SetSize(0);
+		  searchtree->GetIntersecting(box.PMin(),box.PMax(),linenums);
+
+		  for (int j = 0; j < linenums.Size(); j++)
 		    {
-		      BRepGProp::LinearProperties(wexp.Current(), system);
-		      L[i++] = system.Mass();
-		      minL = min (minL, L[i-1]);
+		      int num = linenums[j]-1;
+		      if (i == num) continue;
+		      if ((line.p0-lines[num].p0).Length2() < 1e-15) continue;
+		      if ((line.p0-lines[num].p1).Length2() < 1e-15) continue;
+		      if ((line.p1-lines[num].p0).Length2() < 1e-15) continue;
+		      if ((line.p1-lines[num].p1).Length2() < 1e-15) continue;
+		      mindist = min (mindist, line.Dist(lines[num]));
 		    }
-
-		  if (i == 4 && wexp.More()) continue; // more that 4 edges
-
-		    
-		  double h = minL;
-
-		  for (j = 1; j <= ntriangles; j++)
-		    {
-		      int k;
-		      gp_Pnt p[3];
-		      gp_Pnt2d par[3];
-
-		      for (k = 1; k <=3; k++)
-			{
-			  int n = triangulation->Triangles()(j)(k);
-			  p[k-1] = triangulation->Nodes()(n);
-			  par[k-1] = triangulation->UVNodes()(n);
-			}
-
-		      double lp2p0 = p[2].Distance(p[0]);
-		      double lp2p1 = p[2].Distance(p[1]);
-
-		      prop.SetParameters (p[2].X(), p[2].Y());
 		  
-		      double h0 = h/lp2p0;
-		      double h1 = h/lp2p1;
-
-		      h0 = min (h0, 1.0);
-		      h1 = min (h1, 1.0);
-
-		      double l0, l1, l2;
-		      for (l0 = 0; l0 <= 1; l0+=h0)
-		      for (l1 = 0; l1 <= 1-l0; l1+=h1)
-		      {
-			l2 = 1-l0-l1;
-			double u = l0*par[0].X() + l1*par[1].X() + l2*par[2].X();
-			double v = l0*par[0].Y() + l1*par[1].Y() + l2*par[2].Y();
+		  mindist *= stlparam.resthcloseedgefac;
 
-			prop.SetParameters (u, v);
-			gp_Pnt pnt = prop.Value();
-
-			mesh->RestrictLocalH (Point3d(pnt.X(), pnt.Y(), pnt.Z()), h);
-		      }
-		      
+		  if (mindist < 1e-3) 
+		    {
+		      (*testout) << "extremely small local h: " << mindist
+				 << " --> setting to 1e-3" << endl;
+		      mindist = 1e-3;
 		    }
+
+		  mesh->RestrictLocalHLine(line.p0, line.p1, mindist);
 		}
-	      */
+	    }
+				   
 
 
-	    }
-	  
 	  multithread.task = savetask;
+
 	}
     }
 
@@ -559,6 +1077,67 @@ int OCCGenerateMesh (OCCGeometry & geom,
     {
       FindEdges (geom, *mesh);
 
+      /*
+      cout << "Removing redundant points" << endl;
+      
+      int i, j;
+      int np = mesh->GetNP();
+      ARRAY<int> equalto;
+
+      equalto.SetSize (np);
+      equalto = 0;
+
+      for (i = 1; i <= np; i++)
+	{
+	  for (j = i+1; j <= np; j++)
+	    {
+	      if (!equalto[j-1] && (Dist2 (mesh->Point(i), mesh->Point(j)) < 1e-12))
+		equalto[j-1] = i;
+	    }
+	}
+
+      for (i = 1; i <= np; i++)
+	if (equalto[i-1])
+	  {
+	    cout << "Point " << i << " is equal to Point " << equalto[i-1] << endl;
+	    for (j = 1; j <= mesh->GetNSeg(); j++)
+	      {
+		Segment & seg = mesh->LineSegment(j);
+		if (seg.p1 == i) seg.p1 = equalto[i-1];
+		if (seg.p2 == i) seg.p2 = equalto[i-1];
+	      }
+	  }
+
+      cout << "Removing degenerated segments" << endl;
+      for (j = 1; j <= mesh->GetNSeg(); j++)
+	{
+	  Segment & seg = mesh->LineSegment(j);
+	  if (seg.p1 == seg.p2)
+	    {
+	      mesh->DeleteSegment(j);
+	      cout << "Deleting Segment " << j << endl;
+	    }
+	}
+
+      mesh->Compress();
+      */
+
+      /*
+      for (int i = 1; i <= geom.fmap.Extent(); i++)
+	{
+	  Handle(Geom_Surface) hf1 =
+	    BRep_Tool::Surface(TopoDS::Face(geom.fmap(i)));
+	  for (int j = i+1; j <= geom.fmap.Extent(); j++)
+	    {
+	      Handle(Geom_Surface) hf2 = 
+		BRep_Tool::Surface(TopoDS::Face(geom.fmap(j)));
+	      if (hf1 == hf2) cout << "face " << i << " and face " << j << " lie on same surface" << endl;
+	    }
+	}
+      */
+
+
+
 #ifdef LOG_STREAM      
       (*logout) << "Edges meshed" << endl
 		<< "time = " << GetTime() << " sec" << endl
@@ -629,7 +1208,7 @@ int OCCGenerateMesh (OCCGeometry & geom,
     {
       multithread.task = "Volume optimization";
       
-      OptimizeVolume (mparam, *mesh, NULL);
+      OptimizeVolume (mparam, *mesh);
       if (multithread.terminate) return TCL_OK;
       
 #ifdef STAT_STREAM
diff --git a/Netgen/libsrc/occ/occgeom.cpp b/Netgen/libsrc/occ/occgeom.cpp
index bc50d82975..e24f52fa0a 100644
--- a/Netgen/libsrc/occ/occgeom.cpp
+++ b/Netgen/libsrc/occ/occgeom.cpp
@@ -2,83 +2,756 @@
 
 #include <mystdlib.h>
 #include <occgeom.hpp>  
+#include "ShapeAnalysis_ShapeTolerance.hxx"
+#include "ShapeAnalysis_ShapeContents.hxx"
+#include "ShapeAnalysis_CheckSmallFace.hxx"
+#include "ShapeAnalysis_DataMapOfShapeListOfReal.hxx"
+#include "BRepAlgoAPI_Fuse.hxx"
+#include "BRepCheck_Analyzer.hxx"
+#include "BRepLib.hxx"
+#include "ShapeBuild_ReShape.hxx"
+#include "ShapeFix.hxx"
+#include "ShapeFix_FixSmallFace.hxx"
 
 
 namespace netgen
 {
 
-void PrintContents (OCCGeometry * geom)
+void OCCGeometry :: PrintNrShapes ()
 {
-  TopExp_Explorer exp0;
-  int cnt;
+  TopExp_Explorer e;
+  int count = 0;
+  for (e.Init(shape, TopAbs_COMPSOLID); e.More(); e.Next()) count++;
+  cout << "CompSolids: " << count << endl;
 
-  (*testout) << "OCC CONTENTS" << endl
-	     << "============" << endl;
+  cout << "Solids    : " << somap.Extent() << endl;
+  cout << "Shells    : " << shmap.Extent() << endl;
+  cout << "Faces     : " << fmap.Extent() << endl;
+  cout << "Edges     : " << emap.Extent() << endl;
+  cout << "Vertices  : " << vmap.Extent() << endl;
+}
 
-  for (cnt = 0, exp0.Init(geom->shape, TopAbs_COMPOUND); exp0.More(); exp0.Next()) cnt++;
-  (*testout) << "COMPOUND : " << cnt << endl;
 
-  for (cnt = 0, exp0.Init(geom->shape, TopAbs_COMPSOLID); exp0.More(); exp0.Next()) cnt++;
-  (*testout) << "COMPSOLID: " << cnt << endl;
+void PrintContents (OCCGeometry * geom)
+{
+  ShapeAnalysis_ShapeContents cont;
+  cont.Clear();
+  cont.Perform(geom->shape);
 
-  for (cnt = 0, exp0.Init(geom->shape, TopAbs_SOLID); exp0.More(); exp0.Next()) cnt++;
-  (*testout) << "SOLID    : " << cnt << endl;
+  (*testout) << "OCC CONTENTS" << endl;
+  (*testout) << "============" << endl;
+  (*testout) << "SOLIDS   : " << cont.NbSolids() << endl;
+  (*testout) << "SHELLS   : " << cont.NbShells() << endl;
+  (*testout) << "FACES    : " << cont.NbFaces() << endl;
+  (*testout) << "WIRES    : " << cont.NbWires() << endl;
+  (*testout) << "EDGES    : " << cont.NbEdges() << endl;
+  (*testout) << "VERTICES : " << cont.NbVertices() << endl;
 
-  for (cnt = 0, exp0.Init(geom->shape, TopAbs_SHELL); exp0.More(); exp0.Next()) cnt++;
-  (*testout) << "SHELL    : " << cnt << endl;
+  TopExp_Explorer e;
+  int count = 0;
+  for (e.Init(geom->shape, TopAbs_COMPOUND); e.More(); e.Next())
+    count++;
+  (*testout) << "Compounds: " << count << endl;
 
-  for (cnt = 0, exp0.Init(geom->shape, TopAbs_FACE); exp0.More(); exp0.Next()) cnt++;
-  (*testout) << "FACE     : " << cnt << endl;
+  count = 0;
+  for (e.Init(geom->shape, TopAbs_COMPSOLID); e.More(); e.Next())
+    count++;
+  (*testout) << "CompSolids: " << count << endl;
 
-  for (cnt = 0, exp0.Init(geom->shape, TopAbs_WIRE); exp0.More(); exp0.Next()) cnt++;
-  (*testout) << "WIRE     : " << cnt << endl;
+  (*testout) << endl;
 
-  for (cnt = 0, exp0.Init(geom->shape, TopAbs_EDGE); exp0.More(); exp0.Next()) cnt++;
-  (*testout) << "EDGE     : " << cnt << endl;
+  cout << "Highest entry in topology hierarchy: " << endl;
+  if (count)
+    cout << count << " composite solid(s)" << endl;
+  else
+    if (geom->somap.Extent())
+      cout << geom->somap.Extent() << " solid(s)" << endl;
+    else
+      if (geom->shmap.Extent())
+	cout << geom->shmap.Extent() << " shells(s)" << endl;
+      else
+	if (geom->fmap.Extent())
+	  cout << geom->fmap.Extent() << " face(s)" << endl;
+	else
+	  if (geom->wmap.Extent())
+	    cout << geom->wmap.Extent() << " wire(s)" << endl;
+	  else
+	    if (geom->emap.Extent())
+	      cout << geom->emap.Extent() << " edge(s)" << endl;
+	    else
+	      if (geom->vmap.Extent())
+		cout << geom->vmap.Extent() << " vertices(s)" << endl;
+	      else
+		cout << "no entities" << endl;
 
-  for (cnt = 0, exp0.Init(geom->shape, TopAbs_VERTEX); exp0.More(); exp0.Next()) cnt++;
-  (*testout) << "VERTEX   : " << cnt << endl;
 }
 
 
-void HealGeometry (OCCGeometry * geom)
+
+void OCCGeometry :: HealGeometry ()
 {
-  TopExp_Explorer exp0;
-  int cnt;
+  int nrc = 0, nrcs = 0,
+    nrso = somap.Extent(),
+    nrsh = shmap.Extent(),
+    nrf = fmap.Extent(),
+    nrw = wmap.Extent(),
+    nre = emap.Extent(),
+    nrv = vmap.Extent();
+
+  TopExp_Explorer e;
+  for (e.Init(shape, TopAbs_COMPOUND); e.More(); e.Next()) nrc++;
+  for (e.Init(shape, TopAbs_COMPSOLID); e.More(); e.Next()) nrcs++;
+
+  double surfacecont = 0;
+
+  for (int i = 1; i <= fmap.Extent(); i++)
+    {
+      GProp_GProps system;
+      BRepGProp::LinearProperties(fmap(i), system);
+      surfacecont += system.Mass();
+    }
+
+  cout << "Starting geometry healing procedure (tolerance: " << tolerance << ")" << endl
+       << "-----------------------------------" << endl;
+
+  if (fixsmalledges)
+    {
+      cout << endl << "- fixing small edges" << endl;
+
+      Handle(ShapeFix_Wire) sfw;
+      Handle_ShapeBuild_ReShape rebuild = new ShapeBuild_ReShape;
+      rebuild->Apply(shape);
+
+      for (int i = 1; i <= fmap.Extent(); i++)
+	{
+	  TopExp_Explorer exp1;
+	  for (exp1.Init (fmap(i), TopAbs_WIRE); exp1.More(); exp1.Next())
+	    {
+	      TopoDS_Wire oldwire = TopoDS::Wire(exp1.Current());
+	      sfw = new ShapeFix_Wire (oldwire, TopoDS::Face(fmap(i)),tolerance);
+	      sfw->ModifyTopologyMode() = Standard_True;
+
+	      if (sfw->FixSmall (false, tolerance))
+		{
+		  cout << "Fixed small edge in wire " << wmap.FindIndex (oldwire) << endl;
+		  TopoDS_Wire newwire = sfw->Wire();
+		  rebuild->Replace(oldwire, newwire, Standard_False);
+		}
+	      if ((sfw->StatusSmall(ShapeExtend_FAIL1)) ||
+		  (sfw->StatusSmall(ShapeExtend_FAIL2)) ||
+		  (sfw->StatusSmall(ShapeExtend_FAIL3)))
+		cout << "Failed to fix small edge in wire " << wmap.FindIndex (oldwire) << endl;
+
+	      
+	    }
+	}
+
+      shape = rebuild->Apply(shape);
+
+
+
+      {
+      Handle_ShapeBuild_ReShape rebuild = new ShapeBuild_ReShape;
+      rebuild->Apply(shape);
+      TopExp_Explorer exp1;
+      for (exp1.Init (shape, TopAbs_EDGE); exp1.More(); exp1.Next())
+	{
+	  TopoDS_Edge edge = TopoDS::Edge(exp1.Current());
+	  if (vmap.FindIndex(TopExp::FirstVertex (edge)) == 
+	      vmap.FindIndex(TopExp::LastVertex (edge)))
+	    {
+	      GProp_GProps system;
+	      BRepGProp::LinearProperties(edge, system);
+	      if (system.Mass() < tolerance)
+		{
+		  cout << "removing degenerated edge " << emap.FindIndex(edge) << endl;
+		  rebuild->Remove(edge, false);
+		}
+	    }
+	}
+      shape = rebuild->Apply(shape);
+      }
+
+
+      Handle(ShapeFix_Wireframe) sfwf = new ShapeFix_Wireframe;
+      sfwf->SetPrecision(tolerance);
+      sfwf->Load (shape);
+
+      if (sfwf->FixSmallEdges())
+	{
+	  cout << endl << "- fixing wire frames" << endl;  
+	  if (sfwf->StatusSmallEdges(ShapeExtend_OK)) cout << "no small edges found" << endl;
+	  if (sfwf->StatusSmallEdges(ShapeExtend_DONE1)) cout << "some small edges fixed" << endl;
+	  if (sfwf->StatusSmallEdges(ShapeExtend_FAIL1)) cout << "failed to fix some small edges" << endl;
+	}
+  
+
+      if (sfwf->FixWireGaps())
+	{
+	  cout << endl << "- fixing wire gaps" << endl;
+	  if (sfwf->StatusWireGaps(ShapeExtend_OK)) cout << "no gaps found" << endl;
+	  if (sfwf->StatusWireGaps(ShapeExtend_DONE1)) cout << "some 2D gaps fixed" << endl;
+	  if (sfwf->StatusWireGaps(ShapeExtend_DONE2)) cout << "some 3D gaps fixed" << endl;
+	  if (sfwf->StatusWireGaps(ShapeExtend_FAIL1)) cout << "failed to fix some 2D gaps" << endl;
+	  if (sfwf->StatusWireGaps(ShapeExtend_FAIL2)) cout << "failed to fix some 3D gaps" << endl;
+	}
+      
+
+      shape = sfwf->Shape();
+    }
 
 
-  for (cnt = 0, exp0.Init(geom->shape, TopAbs_SOLID); exp0.More(); exp0.Next()) cnt++;
-  if (cnt == 0)
+
+
+
+  if (fixspotstripfaces)
+    {
+  
+      cout << endl << "- fixing spot and strip faces" << endl;
+      Handle(ShapeFix_FixSmallFace) sffsm = new ShapeFix_FixSmallFace();
+      sffsm -> Init (shape);
+      sffsm -> SetPrecision (tolerance);
+      sffsm -> Perform();
+      
+      shape = sffsm -> FixShape();
+    }
+
+  if (sewfaces)
     {
-      cout << "OCC: Geometry file invalid! No solids. Generating one common solid" << endl;
+      cout << endl << "- sewing faces" << endl;
+
+      TopExp_Explorer exp0;
 
-      BRepOffsetAPI_Sewing sewedObj;
-      for (exp0.Init(geom->shape, TopAbs_FACE); exp0.More(); exp0.Next())
-	sewedObj.Add (TopoDS::Face(exp0.Current()));
+      BRepOffsetAPI_Sewing sewedObj(tolerance);
+
+      for (exp0.Init (shape, TopAbs_FACE); exp0.More(); exp0.Next())
+	{
+	  TopoDS_Face face = TopoDS::Face (exp0.Current());
+	  sewedObj.Add (face);
+	}
+      
       sewedObj.Perform();
+  
+      if (!sewedObj.SewedShape().IsNull())
+	shape = sewedObj.SewedShape();
+      else
+	cout << " not possible";
+    }
+
+  if (makesolids)
+    {  
+      cout << endl << "- making solids" << endl;
+      
+      TopExp_Explorer exp0;
 
       BRepBuilderAPI_MakeSolid ms;
-      for (exp0.Init(sewedObj.SewedShape(), TopAbs_SHELL); exp0.More(); exp0.Next())
-	ms.Add (TopoDS::Shell(exp0.Current()));
+      int count = 0;
+      for (exp0.Init(shape, TopAbs_SHELL); exp0.More(); exp0.Next())
+	{
+	  count++;
+	  ms.Add (TopoDS::Shell(exp0.Current()));
+	}
+      
+      if (!count)
+	{
+	  cout << " not possible (no shells)" << endl;
+	}
+      else
+	{
+	  BRepCheck_Analyzer ba(ms);
+	  if (ba.IsValid ())
+	    {
+	      Handle(ShapeFix_Shape) sfs = new ShapeFix_Shape;
+	      sfs->Init (ms);
+	      sfs->SetPrecision(tolerance);
+	      sfs->SetMaxTolerance(tolerance);
+	      sfs->Perform();
+	      shape = sfs->Shape();
+	      
+	      for (exp0.Init(shape, TopAbs_SOLID); exp0.More(); exp0.Next())
+		{
+		  TopoDS_Solid solid = TopoDS::Solid(exp0.Current());
+		  TopoDS_Solid newsolid = solid;
+		  BRepLib::OrientClosedSolid (newsolid);
+		  Handle_ShapeBuild_ReShape rebuild = new ShapeBuild_ReShape;
+		  //		  rebuild->Apply(shape);
+		  rebuild->Replace(solid, newsolid, Standard_False);
+		  TopoDS_Shape newshape = rebuild->Apply(shape, TopAbs_COMPSOLID, 1);
+		  //		  TopoDS_Shape newshape = rebuild->Apply(shape);
+		  shape = newshape;
+		}
+	    }
+	  else
+	    cout << " not possible" << endl;
+	}
+    }
+
+  BuildFMap();
+
+  double newsurfacecont = 0;
+
+  for (int i = 1; i <= fmap.Extent(); i++)
+    {
+      GProp_GProps system;
+      BRepGProp::LinearProperties(fmap(i), system);
+      newsurfacecont += system.Mass();
+    }
+
+  int nnrc = 0, nnrcs = 0,
+    nnrso = somap.Extent(),
+    nnrsh = shmap.Extent(),
+    nnrf = fmap.Extent(),
+    nnrw = wmap.Extent(),
+    nnre = emap.Extent(),
+    nnrv = vmap.Extent();
+
+  for (e.Init(shape, TopAbs_COMPOUND); e.More(); e.Next()) nnrc++;
+  for (e.Init(shape, TopAbs_COMPSOLID); e.More(); e.Next()) nnrcs++;
+
+  cout << "-----------------------------------" << endl;
+  cout << "Compounds       : " << nnrc << " (" << nrc << ")" << endl;
+  cout << "Composite solids: " << nnrcs << " (" << nrcs << ")" << endl;
+  cout << "Solids          : " << nnrso << " (" << nrso << ")" << endl;
+  cout << "Shells          : " << nnrsh << " (" << nrsh << ")" << endl;
+  cout << "Wires           : " << nnrw << " (" << nrw << ")" << endl;
+  cout << "Faces           : " << nnrf << " (" << nrf << ")" << endl;
+  cout << "Edges           : " << nnre << " (" << nre << ")" << endl;
+  cout << "Vertices        : " << nnrv << " (" << nrv << ")" << endl;
+  cout << endl;
+  cout << "Totol surface area : " << newsurfacecont << " (" << surfacecont << ")" << endl;
+  cout << endl;
+
+}
+ 
+
+
+
+void OCCGeometry :: BuildFMap()
+{
+  somap.Clear();
+  shmap.Clear();
+  fmap.Clear();
+  wmap.Clear();
+  emap.Clear();
+  vmap.Clear();
+  
+  TopExp_Explorer exp0, exp1, exp2, exp3, exp4, exp5;
+  
+  for (exp0.Init(shape, TopAbs_SOLID);
+       exp0.More(); exp0.Next())
+    {
+      TopoDS_Solid solid = TopoDS::Solid (exp0.Current());
+      
+      if (somap.FindIndex(TopoDS::Solid (exp0.Current())) < 1)
+	{
+	  somap.Add (TopoDS::Solid (exp0.Current()));
+	  
+	  for (exp1.Init(exp0.Current(), TopAbs_SHELL);
+	       exp1.More(); exp1.Next())
+	    {
+	      TopoDS_Shell shell = TopoDS::Shell (exp1.Current().Composed (exp0.Current().Orientation()));
+	      if (shmap.FindIndex(shell) < 1)
+		{
+		  shmap.Add (shell);
+		  
+		  for (exp2.Init(shell, TopAbs_FACE);
+		       exp2.More(); exp2.Next())
+		    {
+		      TopoDS_Face face = TopoDS::Face(exp2.Current().Composed(shell.Orientation()));
+		      if (fmap.FindIndex(face) < 1)
+			{
+			  fmap.Add (face);
+			  
+			  for (exp3.Init(exp2.Current(), TopAbs_WIRE);
+			       exp3.More(); exp3.Next())
+			    {
+			      TopoDS_Wire wire = TopoDS::Wire (exp3.Current().Composed(face.Orientation()));
+			      if (wmap.FindIndex(wire) < 1)
+				{
+				  wmap.Add (wire);
+				  
+				  for (exp4.Init(exp3.Current(), TopAbs_EDGE);
+				       exp4.More(); exp4.Next())
+				    {
+				      TopoDS_Edge edge = TopoDS::Edge(exp4.Current().Composed(wire.Orientation()));
+				      if (emap.FindIndex(edge) < 1)
+					{
+					  emap.Add (edge);
+					  for (exp5.Init(exp4.Current(), TopAbs_VERTEX);
+					       exp5.More(); exp5.Next())
+					    {
+					      TopoDS_Vertex vertex = TopoDS::Vertex(exp5.Current());
+					      if (vmap.FindIndex(vertex) < 1)
+						vmap.Add (vertex);
+					    }
+					}
+				    }
+				}
+			    }
+			}
+		    }
+		}
+	    }
+	}
+    }
+  
+  // Free Shells
+  for (exp1.Init(exp0.Current(), TopAbs_SHELL, TopAbs_SOLID);
+       exp1.More(); exp1.Next())
+    {
+      TopoDS_Shape shell = exp1.Current().Composed (exp0.Current().Orientation());
+      if (shmap.FindIndex(shell) < 1)
+	{
+	  shmap.Add (shell);
+	  
+	  for (exp2.Init(shell, TopAbs_FACE);
+	       exp2.More(); exp2.Next())
+	    {
+	      TopoDS_Face face = TopoDS::Face(exp2.Current().Composed(shell.Orientation()));
+	      if (fmap.FindIndex(face) < 1)
+		{
+		  fmap.Add (face);
+		  
+		  for (exp3.Init(exp2.Current(), TopAbs_WIRE);
+		       exp3.More(); exp3.Next())
+		    {
+		      TopoDS_Wire wire = TopoDS::Wire (exp3.Current());
+		      if (wmap.FindIndex(wire) < 1)
+			{
+			  wmap.Add (wire);
+			  
+			  for (exp4.Init(exp3.Current(), TopAbs_EDGE);
+			       exp4.More(); exp4.Next())
+			    {
+			      TopoDS_Edge edge = TopoDS::Edge(exp4.Current());
+			      if (emap.FindIndex(edge) < 1)
+				{
+				  emap.Add (edge);
+				  for (exp5.Init(exp4.Current(), TopAbs_VERTEX);
+				       exp5.More(); exp5.Next())
+				    {
+				      TopoDS_Vertex vertex = TopoDS::Vertex(exp5.Current());
+				      if (vmap.FindIndex(vertex) < 1)
+					vmap.Add (vertex);
+				    }
+				}
+			    }
+			}
+		    }
+		}
+	    }
+	}
+    }
+  
+  
+  // Free Faces
+  
+  for (exp2.Init(shape, TopAbs_FACE, TopAbs_SHELL);
+       exp2.More(); exp2.Next())
+    {
+      TopoDS_Face face = TopoDS::Face(exp2.Current());
+      if (fmap.FindIndex(face) < 1)
+	{
+	  fmap.Add (face);
+	  
+	  for (exp3.Init(exp2.Current(), TopAbs_WIRE);
+	       exp3.More(); exp3.Next())
+	    {
+	      TopoDS_Wire wire = TopoDS::Wire (exp3.Current());
+	      if (wmap.FindIndex(wire) < 1)
+		{
+		  wmap.Add (wire);
+		  
+		  for (exp4.Init(exp3.Current(), TopAbs_EDGE);
+		       exp4.More(); exp4.Next())
+		    {
+		      TopoDS_Edge edge = TopoDS::Edge(exp4.Current());
+		      if (emap.FindIndex(edge) < 1)
+			{
+			  emap.Add (edge);
+			  for (exp5.Init(exp4.Current(), TopAbs_VERTEX);
+			       exp5.More(); exp5.Next())
+			    {
+			      TopoDS_Vertex vertex = TopoDS::Vertex(exp5.Current());
+			      if (vmap.FindIndex(vertex) < 1)
+				vmap.Add (vertex);
+			    }
+			}
+		    }
+		}
+	    }
+	}
+    }
+
+
+  // Free Wires
+  
+  for (exp3.Init(shape, TopAbs_WIRE, TopAbs_FACE);
+       exp3.More(); exp3.Next())
+    {
+      TopoDS_Wire wire = TopoDS::Wire (exp3.Current());
+      if (wmap.FindIndex(wire) < 1)
+	{
+	  wmap.Add (wire);
+	  
+	  for (exp4.Init(exp3.Current(), TopAbs_EDGE);
+	       exp4.More(); exp4.Next())
+	    {
+	      TopoDS_Edge edge = TopoDS::Edge(exp4.Current());
+	      if (emap.FindIndex(edge) < 1)
+		{
+		  emap.Add (edge);
+		  for (exp5.Init(exp4.Current(), TopAbs_VERTEX);
+		       exp5.More(); exp5.Next())
+		    {
+		      TopoDS_Vertex vertex = TopoDS::Vertex(exp5.Current());
+		      if (vmap.FindIndex(vertex) < 1)
+			vmap.Add (vertex);
+		    }
+		}
+	    }
+	}
+    }
+
+
+  // Free Edges
+  
+  for (exp4.Init(shape, TopAbs_EDGE, TopAbs_WIRE);
+       exp4.More(); exp4.Next())
+    {
+      TopoDS_Edge edge = TopoDS::Edge(exp4.Current());
+      if (emap.FindIndex(edge) < 1)
+	{
+	  emap.Add (edge);
+	  for (exp5.Init(exp4.Current(), TopAbs_VERTEX);
+	       exp5.More(); exp5.Next())
+	    {
+	      TopoDS_Vertex vertex = TopoDS::Vertex(exp5.Current());
+	      if (vmap.FindIndex(vertex) < 1)
+		vmap.Add (vertex);
+	    }
+	}
+    }
+
+
+  // Free Vertices
+  
+  for (exp5.Init(shape, TopAbs_VERTEX, TopAbs_EDGE);
+       exp5.More(); exp5.Next())
+    {
+      TopoDS_Vertex vertex = TopoDS::Vertex(exp5.Current());
+      if (vmap.FindIndex(vertex) < 1)
+	vmap.Add (vertex);
+    }
+
+
+
+  
+  facemeshstatus.SetSize (fmap.Extent());
+  facemeshstatus = 0;
+
+  fvispar.SetSize (fmap.Extent());
+  evispar.SetSize (emap.Extent());
+  vvispar.SetSize (vmap.Extent());
+}
+
+
+
+void OCCGeometry :: SewFaces ()
+{
+  (*testout) << "Trying to sew faces ..." << endl;
+  cout << "Trying to sew faces ..." << flush;
+
+  BRepOffsetAPI_Sewing sewedObj(1);
+  //  BRepOffsetAPI_Sewing sewedObj(healingtolerance);
+
+  for (int i = 1; i <= fmap.Extent(); i++)
+    {
+      TopoDS_Face face = TopoDS::Face (fmap(i));
+      sewedObj.Add (face);
+    }
+  
+  sewedObj.Perform();
+  
+  if (!sewedObj.SewedShape().IsNull())
+    {
+      shape = sewedObj.SewedShape();
+      cout << " done" << endl;
+    }
+  else
+    cout << " not possible";
+  
+  /*
+  ShapeUpgrade_ShellSewing sewing;
+  TopoDS_Shape sh = sewing.ApplySewing (shape);
+  shape = sh;
+  */
+}
+
+
+
+
+
+void OCCGeometry :: MakeSolid ()
+{
+  TopExp_Explorer exp0;
+
+  (*testout) << "Trying to build solids ..." << endl;
+  cout << "Trying to build solids ..." << flush;
+
+  BRepBuilderAPI_MakeSolid ms;
+  int count = 0;
+  for (exp0.Init(shape, TopAbs_SHELL); exp0.More(); exp0.Next())
+    {
+      count++;
+      ms.Add (TopoDS::Shell(exp0.Current()));
+    }
+
+  if (!count)
+    {
+     cout << " not possible (no shells)" << endl;
+     return;
+    }
+
+  BRepCheck_Analyzer ba(ms);
+  if (ba.IsValid ())
+    {
+      Handle(ShapeFix_Shape) sfs = new ShapeFix_Shape;
+      sfs->Init (ms);
+  
+      sfs->SetPrecision(1e-5);
+      sfs->SetMaxTolerance(1e-5);
+      
+      sfs->Perform();
 
-      geom->shape = ms;
+      shape = sfs->Shape();
+      
+      for (exp0.Init(shape, TopAbs_SOLID); exp0.More(); exp0.Next())
+	{
+	  TopoDS_Solid solid = TopoDS::Solid(exp0.Current());
+	  TopoDS_Solid newsolid = solid;
+	  BRepLib::OrientClosedSolid (newsolid);
+	  Handle_ShapeBuild_ReShape rebuild = new ShapeBuild_ReShape;
+	  //	  rebuild->Apply(shape);
+	  rebuild->Replace(solid, newsolid, Standard_False);
+	  //	  TopoDS_Shape newshape = rebuild->Apply(shape);
+	  
+	  TopoDS_Shape newshape = rebuild->Apply(shape, TopAbs_SHAPE, 1);
+	  shape = newshape;
+	}
+      
+      cout << " done" << endl;
     }
+  else
+    cout << " not possible" << endl;
+}
+
+
+void OCCGeometry :: BuildVisualizationMesh ()
+{
+  cout << "Preparing visualization (deflection = " << vispar.occdeflection << ") ... " << flush;
+  BRepTools::Clean (shape);
+  BRepMesh_IncrementalMesh::BRepMesh_IncrementalMesh (shape, vispar.occdeflection, true);
+  cout << "done" << endl;
+  
+  Bnd_Box bb;
+  BRepBndLib::Add (shape, bb);
+  
+  double x1,y1,z1,x2,y2,z2;
+  bb.Get (x1,y1,z1,x2,y2,z2);
+  Point<3> p1 = Point<3> (x1,y1,z1);
+  Point<3> p2 = Point<3> (x2,y2,z2);
+  
+  (*testout) << "Bounding Box = [" << p1 << " - " << p2 << "]" << endl;
+  boundingbox = Box<3> (p1,p2);
+  SetCenter();
 }
 
 
+
+  bool OCCGeometry :: FastProject (int surfi, Point<3> & ap, double& u, double& v) const
+  {
+    gp_Pnt p(ap(0), ap(1), ap(2));
+  
+    Handle(Geom_Surface) surface = BRep_Tool::Surface(TopoDS::Face(fmap(surfi)));
+  
+    gp_Pnt x = surface->Value (u,v);
+  
+    if (p.SquareDistance(x) <= sqr(PROJECTION_TOLERANCE)) return true;
+  
+    gp_Vec du, dv;
+  
+    surface->D1(u,v,x,du,dv);
+  
+    int count = 0;
+  
+    gp_Pnt xold;
+    gp_Vec n;
+    double det, lambda, mu;
+  
+    do {
+      count++;
+  
+      n = du^dv;
+  
+      det = Det3 (n.X(), du.X(), dv.X(),
+		  n.Y(), du.Y(), dv.Y(),
+		  n.Z(), du.Z(), dv.Z());
+  
+      if (det < 1e-15) return false; 
+  
+      lambda = Det3 (n.X(), p.X()-x.X(), dv.X(),
+		     n.Y(), p.Y()-x.Y(), dv.Y(),
+		     n.Z(), p.Z()-x.Z(), dv.Z())/det;
+  
+      mu     = Det3 (n.X(), du.X(), p.X()-x.X(),
+		     n.Y(), du.Y(), p.Y()-x.Y(),
+		     n.Z(), du.Z(), p.Z()-x.Z())/det;
+    
+      u += lambda;
+      v += mu;
+  
+      xold = x;
+      surface->D1(u,v,x,du,dv);
+  
+    } while (xold.SquareDistance(x) > sqr(PROJECTION_TOLERANCE) && count < 50);
+
+//    (*testout) << "FastProject count: " << count << endl;
+  
+    if (count == 50) return false;
+  
+    ap = Point<3> (x.X(), x.Y(), x.Z());
+  
+    return true;
+  }
+
+
 OCCGeometry * LoadOCC_IGES (const char * filename)
 {
   OCCGeometry * occgeo;
   occgeo = new OCCGeometry;
   
   IGESControl_Reader reader;
+  
+#ifdef OCC52
+  Standard_Integer stat = reader.ReadFile((char*)filename);
+#else
   Standard_Integer stat = reader.LoadFile((char*)filename);
-  reader.TransferRoots (Standard_False); // Tranlate IGES -> OCC
+  reader.Clear();
+#endif
+  
+#ifdef OCC52
+  reader.TransferRoots(); // Tranlate IGES -> OCC
+#else
+  reader.TransferRoots(Standard_False); // Tranlate IGES -> OCC
+#endif
+
   occgeo->shape = reader.OneShape();
   occgeo->changed = 1;
-  PrintContents (occgeo);
-  HealGeometry (occgeo);
   occgeo->BuildFMap();
+  occgeo->BuildVisualizationMesh();
+  PrintContents (occgeo);
+
   return occgeo;
 }
 
@@ -93,12 +766,335 @@ OCCGeometry * LoadOCC_STEP (const char * filename)
   reader.TransferRoots (); // Tranlate STEP -> OCC
   occgeo->shape = reader.OneShape();
   occgeo->changed = 1;
-  PrintContents (occgeo);
-  HealGeometry (occgeo);
   occgeo->BuildFMap();
+  occgeo->BuildVisualizationMesh();
+  PrintContents (occgeo);
+
   return occgeo;
 }
 
+char * shapesname[] =
+  {" ", "CompSolids", "Solids", "Shells",
+   "Faces", "Wires", "Edges", "Vertices"};
+
+char * shapename[] =
+  {" ", "CompSolid", "Solid", "Shell",
+   "Face", "Wire", "Edge", "Vertex"};
+
+char * orientationstring[] =
+  {"+", "-"};
+
+void OCCGeometry :: RecursiveTopologyTree (const TopoDS_Shape & sh,
+					   stringstream & str,
+					   TopAbs_ShapeEnum l,
+					   bool isfree,
+					   const char * lname)
+{
+  if (l > TopAbs_VERTEX) return;
+
+  TopExp_Explorer e;
+  int count = 0;
+  int count2;
+
+  if (isfree)
+    e.Init(sh, l, TopAbs_ShapeEnum(l-1));
+  else
+    e.Init(sh, l);
+
+  for (; e.More(); e.Next())
+    {
+      count++;
+
+      stringstream lname2;
+      lname2 << lname << "/" << shapename[l] << count;
+      str << lname2.str() << " ";
+
+      switch (e.Current().ShapeType())
+	{
+	case TopAbs_SOLID:
+	  count2 = somap.FindIndex(TopoDS::Solid(e.Current())); break;
+	case TopAbs_SHELL:
+	  count2 = shmap.FindIndex(TopoDS::Shell(e.Current())); break;
+	case TopAbs_FACE:
+	  count2 = fmap.FindIndex(TopoDS::Face(e.Current())); break;
+	case TopAbs_WIRE:
+	  count2 = wmap.FindIndex(TopoDS::Wire(e.Current())); break;
+	case TopAbs_EDGE:
+	  count2 = emap.FindIndex(TopoDS::Edge(e.Current())); break;
+	case TopAbs_VERTEX:
+	  count2 = vmap.FindIndex(TopoDS::Vertex(e.Current())); break;
+	}
+      
+      int nrsubshapes = 0;
+      
+      if (l <= TopAbs_WIRE)
+	{
+	  TopExp_Explorer e2;
+	  for (e2.Init (e.Current(), TopAbs_ShapeEnum (l+1));
+	       e2.More(); e2.Next())
+	    nrsubshapes++;
+	}
+      
+      str << "{" << shapename[l] << " " << count2;
+      
+      if (l <= TopAbs_EDGE)
+	{
+	  str << " (" << orientationstring[e.Current().Orientation()];
+	  if (nrsubshapes != 0) str << ", " << nrsubshapes;
+	  str << ") } ";
+	}
+      else
+	str << " } ";
+
+      RecursiveTopologyTree (e.Current(), str, TopAbs_ShapeEnum (l+1),
+			     false, (char*)lname2.str().c_str());
+      
+    }
+}
+
+void OCCGeometry :: GetTopologyTree (stringstream & str)
+{
+  cout << "Building topology tree ... " << flush;
+  RecursiveTopologyTree (shape, str, TopAbs_COMPSOLID, false, "CompSolids");
+  RecursiveTopologyTree (shape, str, TopAbs_SOLID, true, "FreeSolids");
+  RecursiveTopologyTree (shape, str, TopAbs_SHELL, true, "FreeShells");
+  RecursiveTopologyTree (shape, str, TopAbs_FACE, true, "FreeFaces");
+  RecursiveTopologyTree (shape, str, TopAbs_WIRE, true, "FreeWires");
+  RecursiveTopologyTree (shape, str, TopAbs_EDGE, true, "FreeEdges");
+  RecursiveTopologyTree (shape, str, TopAbs_VERTEX, true, "FreeVertices");
+  str << flush;
+  //  cout << "done" << endl;
+}
+
+void OCCGeometry :: CheckIrregularEntities(stringstream & str)
+{
+  ShapeAnalysis_CheckSmallFace csm;
+
+  csm.SetTolerance (1e-6);
+
+  TopTools_DataMapOfShapeListOfShape mapEdges;
+  ShapeAnalysis_DataMapOfShapeListOfReal mapParam;
+  TopoDS_Compound theAllVert;
+
+  int spotfaces = 0;
+  int stripsupportfaces = 0;
+  int singlestripfaces = 0;
+  int stripfaces = 0;
+  int facessplitbyvertices = 0;
+  int stretchedpinfaces = 0;
+  int smoothpinfaces = 0;
+  int twistedfaces = 0;
+  int edgessamebutnotidentified = 0;
+
+  cout << "checking faces ... " << flush;
+
+  int i;
+  for (i = 1; i <= fmap.Extent(); i++)
+    {
+      TopoDS_Face face = TopoDS::Face (fmap(i));
+      TopoDS_Edge e1, e2;
+
+      if (csm.CheckSpotFace (face))
+	{
+	  if (!spotfaces++)
+	    str << "SpotFace {Spot face} ";
+
+	  (*testout) << "Face " << i << " is a spot face" << endl;
+	  str << "SpotFace/Face" << i << " ";
+	  str << "{Face " << i << " } ";
+	}
+
+      if (csm.IsStripSupport (face))
+	{
+	  if (!stripsupportfaces++)
+	    str << "StripSupportFace {Strip support face} ";
+
+	  (*testout) << "Face " << i << " has strip support" << endl;
+	  str << "StripSupportFace/Face" << i << " ";
+	  str << "{Face " << i << " } ";
+	}
+
+      if (csm.CheckSingleStrip(face, e1, e2))
+	{
+	  if (!singlestripfaces++)
+	    str << "SingleStripFace {Single strip face} ";
+
+	  (*testout) << "Face " << i << " is a single strip (edge " << emap.FindIndex(e1)
+	       << " and edge " << emap.FindIndex(e2) << " are identical)" << endl;
+	  str << "SingleStripFace/Face" << i << " ";
+	  str << "{Face " << i << " (edge " << emap.FindIndex(e1)
+	       << " and edge " << emap.FindIndex(e2) << " are identical)} ";
+	}
+
+      if (csm.CheckStripFace(face, e1, e2))
+	{
+	  if (!stripfaces++)
+	    str << "StripFace {Strip face} ";
+
+	  (*testout) << "Face " << i << " is a strip (edge " << emap.FindIndex(e1)
+		     << " and edge " << emap.FindIndex(e2)
+		     << " are identical)" << endl;
+	  str << "StripFace/Face" << i << " ";
+	  str << "{Face " << i << " (edge " << emap.FindIndex(e1)
+	      << " and edge " << emap.FindIndex(e2) << " are identical)} ";
+	}
+
+      if (int count = csm.CheckSplittingVertices(face, mapEdges, mapParam, theAllVert))
+	{
+	  if (!facessplitbyvertices++)
+	    str << "FaceSplitByVertices {Face split by vertices} ";
+
+	  (*testout) << "Face " << i << " is split by " << count
+		     << " vertex/vertices " << endl;
+	  str << "FaceSplitByVertices/Face" << i << " ";
+	  str << "{Face " << i << " (split by " << count << "vertex/vertices)} ";
+	}
+      
+      int whatrow, sens;
+      if (int type = csm.CheckPin (face, whatrow, sens))
+	{
+	  if (type == 1)
+	    {
+	      if (!smoothpinfaces++)
+		str << "SmoothPinFace {Smooth pin face} ";
+
+	      (*testout) << "Face " << i << " is a smooth pin" << endl;
+	      str << "SmoothPinFace/Face" << i << " ";
+	      str << "{Face " << i << " } ";
+	    }
+	  else
+	    {
+	      if (!stretchedpinfaces++)
+		str << "StretchedPinFace {Stretched pin face} ";
+
+	      (*testout) << "Face " << i << " is a streched pin" << endl;
+	      str << "StretchedPinFace/Face" << i << " ";
+	      str << "{Face " << i << " } ";
+	    }
+	}
+
+      double paramu, paramv;
+      if (csm.CheckTwisted (face, paramu, paramv))
+	{
+	  if (!twistedfaces++)
+	    str << "TwistedFace {Twisted face} ";
+
+	  (*testout) << "Face " << i << " is twisted" << endl;
+	  str << "TwistedFace/Face" << i << " ";
+	  str << "{Face " << i << " } ";
+	}
+    }
+
+  cout << "done" << endl;
+  cout << "checking edges ... " << flush;
+
+  double dmax;
+  int cnt = 0;
+  ARRAY <double> edgeLengths;
+  ARRAY <int> order;
+  edgeLengths.SetSize (emap.Extent());
+  order.SetSize (emap.Extent());
+
+  for (i = 1; i <= emap.Extent(); i++)
+    {
+      TopoDS_Edge edge1 = TopoDS::Edge (emap(i));
+      GProp_GProps system;
+      BRepGProp::LinearProperties(edge1, system);
+      edgeLengths[i-1] = system.Mass();
+      /*
+      int j;
+      for (j = i+1; j <= emap.Extent(); j++)
+	{
+	  TopoDS_Edge edge2 = TopoDS::Edge (emap(j));
+
+	  if (csm.CheckStripEdges(edge1, edge2, csm.Tolerance(), dmax))
+	    {
+	      if (!edgessamebutnotidentified++)
+		str << "EdgesSameButNotIdentified {Edges same but not identified} ";
+
+	      cnt++;
+	      (*testout) << "Edge " << i << " and edge " << j
+			 << " are on one strip (same but not identified)" << endl;
+	      str << "EdgesSameButNotIdentified/Edge" << cnt << " ";
+	      str << "{Edge " << i << " and Edge " << j << "} ";
+	    }
+	}
+      */
+    }
+
+  Sort (edgeLengths, order);
+
+  str << "ShortestEdges {Shortest edges} ";
+  for (i = 1; i <= min(20, emap.Extent()); i++)
+    {
+      str << "ShortestEdges/Edge" << i;
+      str << " {Edge " << order[i-1] << " (L=" << edgeLengths[order[i-1]-1] << ")} ";
+    }
+
+  str << flush;
+
+  cout << "done" << endl;
+
+  /*
+  for (i = 1; i <= shmap.Extent(); i++)
+    {
+      TopoDS_Shell shell = TopoDS::Shell (shmap(i));
+      if (!shell.Closed()) 
+	cout << "Shell " << i << " is not closed" << endl;
+      if (shell.Infinite()) 
+	cout << "Shell " << i << " is infinite" << endl;
+
+      BRepCheck_Analyzer ba(shell);
+      if (!ba.IsValid ())
+	cout << "Shell " << i << " is not valid" << endl;
+    }
+
+  for (i = 1; i <= somap.Extent(); i++)
+    {
+      TopoDS_Solid solid = TopoDS::Solid (somap(i));
+      if (!solid.Closed()) 
+	cout << "Solid " << i << " is not closed" << endl;
+      if (solid.Infinite()) 
+	cout << "Solid " << i << " is infinite" << endl;
+
+      BRepCheck_Analyzer ba(solid);
+      if (!ba.IsValid ())
+	cout << "Solid " << i << " is not valid" << endl;
+    }
+  */
+
+
+}
+
+
+void OCCGeometry :: GetUnmeshedFaceInfo (stringstream & str)
+{
+  for (int i = 1; i <= fmap.Extent(); i++)
+    {
+      if (facemeshstatus[i-1] == -1)
+	str << "Face" << i << " {Face " << i << " } ";
+    }
+  str << flush;
+}
+
+void OCCGeometry :: GetNotDrawableFaces (stringstream & str)
+{
+  for (int i = 1; i <= fmap.Extent(); i++)
+    {
+      if (!fvispar[i-1].IsDrawable())
+	str << "Face" << i << " {Face " << i << " } ";
+    }
+  str << flush;
+}
+
+bool OCCGeometry :: ErrorInSurfaceMeshing ()
+{
+  for (int i = 1; i <= fmap.Extent(); i++)
+    if (facemeshstatus[i-1] == -1)
+      return true;
+
+  return false;
+}
 
 }
 
diff --git a/Netgen/libsrc/occ/occgeom.hpp b/Netgen/libsrc/occ/occgeom.hpp
index 99472e71ff..5082ab3b91 100644
--- a/Netgen/libsrc/occ/occgeom.hpp
+++ b/Netgen/libsrc/occ/occgeom.hpp
@@ -35,6 +35,7 @@
 #include "TopExp.hxx"
 #include "gp_Pnt.hxx"
 #include "TopoDS.hxx"
+#include "TopoDS_Solid.hxx"
 #include "TopExp_Explorer.hxx"
 #include "BRep_Tool.hxx"
 #include "Geom_Curve.hxx"
@@ -51,8 +52,10 @@
 #include "BRepBuilderAPI_MakeShell.hxx"
 #include "BRepBuilderAPI_MakeSolid.hxx"
 #include "BRepOffsetAPI_Sewing.hxx"
+#include "BRepLProp_CLProps.hxx"
 #include "BRepLProp_SLProps.hxx"
 #include "BRepAdaptor_Surface.hxx"
+#include "BRepAdaptor_Curve.hxx"
 #include "Poly_Triangulation.hxx"
 #include "Poly_Array1OfTriangle.hxx"
 #include "TColgp_Array1OfPnt2d.hxx"
@@ -63,114 +66,209 @@
 #include "STEPControl_Reader.hxx"
 #include "TopoDS_Shape.hxx"
 #include "TopoDS_Face.hxx"
-
-
+#include "IGESToBRep_Reader.hxx"
+#include "Interface_Static.hxx"
+#include "GeomAPI_ExtremaCurveCurve.hxx"
+#include "Standard_ErrorHandler.hxx"
+#include "Standard_Failure.hxx"
+#include "ShapeUpgrade_ShellSewing.hxx"
+#include "ShapeFix_Shape.hxx"
+#include "ShapeFix_Wireframe.hxx"
+#include "BRepMesh.hxx"
+#include "BRepMesh_IncrementalMesh.hxx"
+#include "BRepBndLib.hxx"
+#include "Bnd_Box.hxx"
+#include "ShapeAnalysis.hxx"
+#include "ShapeBuild_ReShape.hxx"
+#include "IGESControl_Writer.hxx"
+#include "STEPControl_Writer.hxx"
+#include "StlAPI_Writer.hxx"
+#include "STEPControl_StepModelType.hxx"
 
 namespace netgen
 {
 
+#include "../visualization/vispar.hpp"
+  //  class VisualizationParameters;
+  //  extern VisualizationParameters vispar;
+
+
 #include "occmeshsurf.hpp"
 
+#define PROJECTION_TOLERANCE 1e-10
+
+
+#define ENTITYISVISIBLE 1
+#define ENTITYISHIGHLIGHTED 2
+#define ENTITYISDRAWABLE 4
+
+class EntityVisualizationCode
+{
+  int code;
+
+public:
+
+  EntityVisualizationCode()
+  { code = ENTITYISVISIBLE + !ENTITYISHIGHLIGHTED + ENTITYISDRAWABLE; }
+
+  int IsVisible ()
+  { return code & ENTITYISVISIBLE; }
+
+  int IsHighlighted ()
+  { return code & ENTITYISHIGHLIGHTED; }
+
+  int IsDrawable ()
+  { return code & ENTITYISDRAWABLE; }
+
+  void Show ()
+  { code |= ENTITYISVISIBLE; }
+
+  void Hide ()
+  { code &= ~ENTITYISVISIBLE; }
+
+  void Highlight ()
+  { code |= ENTITYISHIGHLIGHTED; }
+
+  void Lowlight ()
+  { code &= ~ENTITYISHIGHLIGHTED; }
+
+  void SetDrawable ()
+  { code |= ENTITYISDRAWABLE; }
+
+  void SetNotDrawable ()
+  { code &= ~ENTITYISDRAWABLE; }
+};
+
+
+
+inline double Det3 (double a00, double a01, double a02,
+		    double a10, double a11, double a12,
+		    double a20, double a21, double a22)
+{
+  return a00*a11*a22 + a01*a12*a20 + a10*a21*a02 - a20*a11*a02 - a10*a01*a22 - a21*a12*a00;
+}
+
+
+
+#define OCCGEOMETRYVISUALIZATIONNOCHANGE   0
+#define OCCGEOMETRYVISUALIZATIONFULLCHANGE 1
+  // == compute transformation matrices and redraw
+#define OCCGEOMETRYVISUALIZATIONHALFCHANGE 2
+  // == redraw
 
 class OCCGeometry
 {
+  Point<3> center;
+
 public:
   TopoDS_Shape shape;
-  TopTools_IndexedMapOfShape fmap, emap, vmap;
-  double maxsize;
+  TopTools_IndexedMapOfShape fmap, emap, vmap, somap, shmap, wmap;
+  Box<3> boundingbox;
+
+  int changed; 
+  ARRAY<int> facemeshstatus;
+
+  ARRAY<EntityVisualizationCode> fvispar, evispar, vvispar;
 
-  bool changed;
+  double tolerance;
+  bool fixsmalledges;
+  bool fixspotstripfaces;
+  bool sewfaces;
+  bool makesolids;
 
 
   OCCGeometry()
   {
+    somap.Clear();
+    shmap.Clear();
     fmap.Clear();
+    wmap.Clear();
     emap.Clear();
     vmap.Clear();
   }
 
 
-  void BuildFMap()
+  void BuildFMap();
+
+  Box<3> GetBoundingBox()
+  { return boundingbox; }
+
+  int NrSolids()
+  { return somap.Extent(); }
+
+  void SetCenter()
+  { center = boundingbox.Center(); }
+
+  Point<3> Center()
+  { return center; }
+
+  void Project (int surfi, Point<3> & p) const
   {
-    TopExp_Explorer exp0, exp1, exp2, exp3;
-    maxsize = 0;
+    static int cnt = 0;
+    if (++cnt % 1000 == 0) cout << "Project cnt = " << cnt << endl;
 
-    /*    
-    for (exp3.Init(shape, TopAbs_EDGE); exp3.More(); exp3.Next())
+    gp_Pnt pnt(p(0), p(1), p(2));
+
+    GeomAPI_ProjectPointOnSurf proj(pnt, BRep_Tool::Surface(TopoDS::Face(fmap(surfi))));
+    if (proj.NbPoints() == 0)
       {
-	TopoDS_Edge edge = TopoDS::Edge(exp3.Current());
-	(*testout) << edge.Orientation() << endl;
-	if (emap.FindIndex(edge) < 1)
-	  emap.Add (edge);
+	cout << "Projection fails" << endl;
+      }
+    else
+      {
+	pnt = proj.NearestPoint();
+	p = Point<3> (pnt.X(), pnt.Y(), pnt.Z());
       }
-    */
-   
-
-    for (exp0.Init(shape, TopAbs_SOLID); exp0.More(); exp0.Next())
-      for (exp1.Init(exp0.Current(), TopAbs_SHELL); exp1.More(); exp1.Next())
-	{
-	  TopoDS_Shape shell = exp1.Current().Composed (exp0.Current().Orientation());
-
-	  for (exp2.Init(shell, TopAbs_FACE); exp2.More(); exp2.Next())
-	    {
-	      fmap.Add (exp2.Current().Composed(shell.Orientation()));
-	      
-	      for (exp3.Init(exp2.Current(), TopAbs_EDGE); exp3.More(); exp3.Next())
-		{
-		  TopoDS_Edge edge = TopoDS::Edge(exp3.Current());
-		  if (emap.FindIndex(edge) < 1)
-		    emap.Add (edge);
-		  /*
-		  else
-		    {
-		      cout << edge.Orientation() << " = " << emap(emap.FindIndex(edge)).Orientation() << endl;
-		    }
-		  */
-		}
-	      
-	      for (exp3.Init(exp2.Current(), TopAbs_VERTEX); exp3.More(); exp3.Next())
-		{
-		  TopoDS_Vertex vertex = TopoDS::Vertex(exp3.Current());
-		  if (vmap.FindIndex(vertex) < 1)
-		    {
-		      vmap.Add (vertex);
-		      
-		      gp_Pnt p = BRep_Tool::Pnt(vertex);
-		      maxsize = max (maxsize, fabs(p.X()));
-		      maxsize = max (maxsize, fabs(p.Y()));
-		      maxsize = max (maxsize, fabs(p.Y()));
-		    }
-		}
-	    }
-	}
-    maxsize *= 2;
   }
 
+  bool FastProject (int surfi, Point<3> & ap, double& u, double& v) const;
 
-  double MaxSize()
+ 
+  OCCSurface GetSurface (int surfi)
   {
-    return maxsize;
+    cout << "OCCGeometry::GetSurface using PLANESPACE" << endl;
+    return OCCSurface (TopoDS::Face(fmap(surfi)), PLANESPACE);
   }
+  
 
+  void BuildVisualizationMesh ();
 
+  void RecursiveTopologyTree (const TopoDS_Shape & sh,
+			      stringstream & str,
+			      TopAbs_ShapeEnum l,
+			      bool free,
+			      const char * lname);
 
-  void Project (int surfi, Point<3> & p) const
-  {
-    static int cnt = 0;
-    if (cnt++ % 1000 == 0) cout << "Project cnt = " << cnt << endl;
+  void GetTopologyTree (stringstream & str);
 
-    gp_Pnt pnt(p(0), p(1), p(2));
+  void PrintNrShapes ();
 
-    GeomAPI_ProjectPointOnSurf proj(pnt, BRep_Tool::Surface(TopoDS::Face(fmap(surfi))));
-    pnt = proj.NearestPoint();
-    p = Point<3> (pnt.X(), pnt.Y(), pnt.Z());
+  void CheckIrregularEntities (stringstream & str);
+
+  void SewFaces();
+
+  void MakeSolid();
+
+  void HealGeometry();
+
+  void LowLightAll()
+  {
+    for (int i = 1; i <= fmap.Extent(); i++)
+      fvispar[i-1].Lowlight();
+    for (int i = 1; i <= emap.Extent(); i++)
+      evispar[i-1].Lowlight();
+    for (int i = 1; i <= vmap.Extent(); i++)
+      vvispar[i-1].Lowlight();
   }
 
+  void GetUnmeshedFaceInfo (stringstream & str);
+  void GetNotDrawableFaces (stringstream & str);
+  bool ErrorInSurfaceMeshing ();
 };
 
 
 void PrintContents (OCCGeometry * geom);
-void HealGeometry (OCCGeometry * geom);
+
 OCCGeometry * LoadOCC_IGES (const char * filename);
 OCCGeometry * LoadOCC_STEP (const char * filename);
 
diff --git a/Netgen/libsrc/occ/occmeshsurf.cpp b/Netgen/libsrc/occ/occmeshsurf.cpp
index c8be2c0770..034571c9ae 100644
--- a/Netgen/libsrc/occ/occmeshsurf.cpp
+++ b/Netgen/libsrc/occ/occmeshsurf.cpp
@@ -10,12 +10,24 @@ namespace netgen
 {
 #include "occmeshsurf.hpp"
 
+
 void OCCSurface :: GetNormalVector (const Point<3> & p, 
 				    const PointGeomInfo & geominfo,
 				    Vec<3> & n) const
 {
   gp_Pnt pnt;
   gp_Vec du, dv;
+
+  /*
+  double gu = geominfo.u;
+  double gv = geominfo.v;
+
+  if (fabs (gu) < 1e-3) gu = 0;
+  if (fabs (gv) < 1e-3) gv = 0;
+
+  occface->D1(gu,gv,pnt,du,dv);
+  */
+
   occface->D1(geominfo.u,geominfo.v,pnt,du,dv);
 
   n = Cross (Vec<3>(du.X(), du.Y(), du.Z()),
@@ -23,6 +35,7 @@ void OCCSurface :: GetNormalVector (const Point<3> & p,
   n.Normalize();
 
   if (orient == TopAbs_REVERSED) n = -1*n;
+  //  (*testout) << "GetNormalVector" << endl;
 }
 
 
@@ -31,40 +44,127 @@ void OCCSurface :: DefineTangentialPlane (const Point<3> & ap1,
 					  const Point<3> & ap2,
 					  const PointGeomInfo & geominfo2)
 {
-  p1 = ap1; p2 = ap2;
+  if (projecttype == PLANESPACE)
+    {
+      p1 = ap1; p2 = ap2;
+
+      //      cout << "p1 = " << p1 << endl;
+      //      cout << "p2 = " << p2 << endl;
+      
+      GetNormalVector (p1, geominfo1, ez);
+      
+      ex = p2 - p1;
+      ex -= (ex * ez) * ez;
+      ex.Normalize();
+      ey = Cross (ez, ex); 
+
+      GetNormalVector (p2, geominfo2, n2);
   
-  GetNormalVector (p1, geominfo1, ez);
+      nmid = 0.5*(n2+ez);
+      
+      ez = nmid;
+      ez.Normalize(); 
+      
+      ex = (p2 - p1).Normalize();
+      ez -= (ez * ex) * ex;
+      ez.Normalize();
+      ey = Cross (ez, ex);
+      nmid = ez;
+    }
+  else
+    {
+      if ( (geominfo1.u < umin) ||
+	   (geominfo1.u > umax) ||
+	   (geominfo2.u < umin) ||
+	   (geominfo2.u > umax) ||
+	   (geominfo1.v < vmin) ||
+	   (geominfo1.v > vmax) ||
+	   (geominfo2.v < vmin) ||
+	   (geominfo2.v > vmax) ) throw UVBoundsException();
+	  
+
+      p1 = ap1; p2 = ap2;
+      psp1 = Point<2>(geominfo1.u, geominfo1.v);
+      psp2 = Point<2>(geominfo2.u, geominfo2.v);
+      
+      Vec<3> n;
+      GetNormalVector (p1, geominfo1, n);
 
-  ex = p2 - p1;
-  ex -= (ex * ez) * ez;
-  ex.Normalize();
-  ey = Cross (ez, ex); 
+      gp_Pnt pnt;
+      gp_Vec du, dv;
+      occface->D1 (geominfo1.u, geominfo1.v, pnt, du, dv);
 
-  /*
-  double u, v;
-  gp_Vec du, dv;
-  gp_Pnt pnt(ap2(0), ap2(1), ap2(2));
-  GeomAPI_ProjectPointOnSurf proj(pnt, occface);
-  proj.LowerDistanceParameters (u, v);
-  occface->D1(u, v, pnt, du, dv);
-  n2 = Cross (Vec<3>(du.X(), du.Y(), du.Z()),
-	      Vec<3>(dv.X(), dv.Y(), dv.Z()));
-  n2.Normalize();
-  if (orient == TopAbs_REVERSED) n2 = -1*n2;
-  */
+      DenseMatrix D1(3,2), D1T(2,3), DDTinv(2,2);
+      D1(0,0) = du.X(); D1(1,0) = du.Y(); D1(2,0) = du.Z();
+      D1(0,1) = dv.X(); D1(1,1) = dv.Y(); D1(2,1) = dv.Z();
 
-  GetNormalVector (p2, geominfo2, n2);
- 
-  nmid = 0.5*(n2+ez);
+      /*
+      (*testout) << "DefineTangentialPlane" << endl
+		 << "---------------------" << endl;
+      (*testout) << "D1 = " << endl << D1 << endl;
+      */
 
-  ez = nmid;
-  ez.Normalize(); 
+      Transpose (D1, D1T);
+      DenseMatrix D1TD1(3,3);
 
-  ex = (p2 - p1).Normalize();
-  ez -= (ez * ex) * ex;
-  ez.Normalize();
-  ey = Cross (ez, ex);
-  nmid = ez;
+      D1TD1 = D1T*D1;
+      if (D1TD1.Det() == 0) throw SingularMatrixException();
+      
+      CalcInverse (D1TD1, DDTinv);
+      DenseMatrix Y(3,2);
+      Vec<3> y1 = (ap2-ap1).Normalize();
+      Vec<3> y2 = Cross(n, y1).Normalize();
+      for (int i = 0; i < 3; i++)
+	{
+	  Y(i,0) = y1(i);
+	  Y(i,1) = y2(i);
+	}
+
+      DenseMatrix A(2,2);
+      A = DDTinv * D1T * Y;
+      DenseMatrix Ainv(2,2);
+
+      if (A.Det() == 0) throw SingularMatrixException();
+
+      CalcInverse (A, Ainv);
+
+      for (int i = 0; i < 2; i++)
+	for (int j = 0; j < 2; j++)
+	  {
+	    Amat(i,j) = A(i,j);
+	    Amatinv(i,j) = Ainv(i,j);
+	  }
+
+      Vec<2> temp = Amatinv * (psp2-psp1);
+      
+
+      double r = temp.Length();
+      //      double alpha = -acos (temp(0)/r);
+      double alpha = -atan2 (temp(1),temp(0));
+      DenseMatrix R(2,2);
+      R(0,0) = cos (alpha);
+      R(1,0) = -sin (alpha);
+      R(0,1) = sin (alpha);
+      R(1,1) = cos (alpha);
+
+
+      A = A*R;
+
+      if (A.Det() == 0) throw SingularMatrixException();
+
+      CalcInverse (A, Ainv);
+    
+
+      for (int i = 0; i < 2; i++)
+	for (int j = 0; j < 2; j++)
+	  {
+	    Amat(i,j) = A(i,j);
+	    Amatinv(i,j) = Ainv(i,j);
+	  }
+
+      temp = Amatinv * (psp2-psp1);
+      
+    };
  
 }
 
@@ -74,42 +174,30 @@ void OCCSurface :: ToPlane (const Point<3> & p3d,
 			    Point<2> & pplane, 
 			    double h, int & zone) const
 {
-  Vec<3> p1p, n;
-  GetNormalVector (p3d, geominfo, n);
-
-  p1p = p3d - p1;
-  pplane(0) = (p1p * ex) / h;
-  pplane(1) = (p1p * ey) / h;
-
-  if (n * nmid < 0)
-    zone = -1;
-  else
-    zone = 0;
-  
-  /*
-  Vec<3> v = p3d - p1;
-  Vec<3> n;
-  GetNormalVector (p3d, geominfo, n);
-
-  if (n * nmid < 0)
-    zone = -1;
+  if (projecttype == PLANESPACE)
+    {
+      Vec<3> p1p, n;
+      GetNormalVector (p3d, geominfo, n);
+      
+      p1p = p3d - p1;
+      pplane(0) = (p1p * ex) / h;
+      pplane(1) = (p1p * ey) / h;
+      
+      if (n * nmid < 0)
+	zone = -1;
+      else
+	zone = 0;
+    }
   else
     {
-      double nom = h*(n(0)*(ex(1)*ey(2)-ex(2)*ey(1)) + 
-		      n(1)*(ex(2)*ey(0)-ex(0)*ey(2)) +
-		      n(2)*(ex(0)*ey(1)-ex(1)*ey(0)));
-
-      pplane(0) = (n(0)*(ey(2)*v(1)-ey(1)*v(2)) +
-		   n(1)*(ey(0)*v(2)-ey(2)*v(0)) +
-		   n(2)*(ey(1)*v(0)-ey(0)*v(1)))/nom;
+      pplane = Point<2>(geominfo.u, geominfo.v);
+      //      (*testout) << "(u,v) = " << geominfo.u << ", " << geominfo.v << endl;
+      pplane = Point<2> (1/h * (Amatinv * (pplane-psp1)));
+      //      pplane = Point<2> (h * (Amatinv * (pplane-psp1)));
+      //      pplane = Point<2> (1/h * ((pplane-psp1)));
 
-      pplane(1) = (n(0)*(ex(1)*v(2)-ex(2)*v(1)) +
-		   n(1)*(ex(2)*v(0)-ex(0)*v(2)) +
-		   n(2)*(ex(0)*v(1)-ex(1)*v(0)))/nom;
-      
       zone = 0;
-    }
-  */
+    };
 }	
 
 
@@ -118,8 +206,25 @@ void OCCSurface :: FromPlane (const Point<2> & pplane,
 			      PointGeomInfo & gi,
 			      double h) 
 { 
-  p3d = p1 + (h * pplane(0)) * ex + (h * pplane(1)) * ey;
-  Project (p3d, gi);  
+  if (projecttype == PLANESPACE)
+    {
+      //      cout << "2d   : " << pplane << endl;
+      p3d = p1 + (h * pplane(0)) * ex + (h * pplane(1)) * ey;
+      //      cout << "3d   : " << p3d << endl;
+      Project (p3d, gi);  
+      //      cout << "proj : " << p3d << endl;
+    }
+  else
+    {
+      //      Point<2> pspnew = Point<2>(1/h * (Amat * Vec<2>(pplane)) + Vec<2>(psp1));
+      Point<2> pspnew = Point<2>(h * (Amat * Vec<2>(pplane)) + Vec<2>(psp1));
+      //      Point<2> pspnew = Point<2>(h * (Vec<2>(pplane)) + Vec<2>(psp1));
+      gi.u = pspnew(0);
+      gi.v = pspnew(1);
+      gi.trignum = 1;
+      gp_Pnt val = occface->Value (gi.u, gi.v);
+      p3d = Point<3> (val.X(), val.Y(), val.Z());
+    };
 }
 
 
@@ -130,7 +235,28 @@ void OCCSurface :: Project (Point<3> & p, PointGeomInfo & gi)
   //  if (cnt++ % 1000 == 0) cout << "********************************************** OCCSurfce :: Project, cnt = " << cnt << endl;
   
   gp_Pnt pnt(p(0), p(1), p(2));
-  GeomAPI_ProjectPointOnSurf proj(pnt, occface);
+
+  //  cout << "pnt = " << pnt.X() << ", " << pnt.Y() << ", " << pnt.Z() << endl;
+
+  GeomAPI_ProjectPointOnSurf proj(pnt, occface, umin, umax, vmin, vmax);
+
+  if (!proj.NbPoints())
+    {
+      cout << "Project Point on Surface FAIL" << endl;
+      throw UVBoundsException();
+    }
+
+  
+  /*
+  cout << "NP = " << proj.NbPoints() << endl;
+
+  for (int i = 1; i <= proj.NbPoints(); i++)
+    {
+      gp_Pnt pnt2 = proj.Point(i);
+      Point<3> p2 = Point<3> (pnt2.X(), pnt2.Y(), pnt2.Z());
+      cout << i << ". p = " << p2 << ", dist = " << (p2-p).Length() << endl;
+    }
+  */
 
   pnt = proj.NearestPoint();
   proj.LowerDistanceParameters (gi.u, gi.v);
@@ -141,8 +267,8 @@ void OCCSurface :: Project (Point<3> & p, PointGeomInfo & gi)
 
 
 Meshing2OCCSurfaces :: Meshing2OCCSurfaces (const TopoDS_Shape & asurf,
-					    const Box<3> & abb)
-  : Meshing2(Box3d(abb.PMin(), abb.PMax())), surface(TopoDS::Face(asurf))
+					    const Box<3> & abb, int aprojecttype)
+  : Meshing2(Box3d(abb.PMin(), abb.PMax())), surface(TopoDS::Face(asurf), aprojecttype)
 {
   ;
 }
@@ -154,7 +280,7 @@ void Meshing2OCCSurfaces :: DefineTransformation (Point3d & p1, Point3d & p2,
 {
   ((OCCSurface&)surface).DefineTangentialPlane (p1, *geominfo1, p2, *geominfo2);
 }
-
+ 
 void Meshing2OCCSurfaces :: TransformToPlain (const Point3d & locpoint, 
 					   const MultiPointGeomInfo & geominfo,
 					   Point2d & planepoint, 
@@ -208,7 +334,7 @@ void MeshOptimize2dOCCSurfaces :: ProjectPoint2 (INDEX surfind, INDEX surfind2,
 					      Point3d & p) const
 {
   TopExp_Explorer exp0, exp1;
-  int done = 0;
+  bool done = false;
   Handle(Geom_Curve) c;
 
   for (exp0.Init(geometry.fmap(surfind), TopAbs_EDGE); !done && exp0.More(); exp0.Next())
@@ -216,7 +342,7 @@ void MeshOptimize2dOCCSurfaces :: ProjectPoint2 (INDEX surfind, INDEX surfind2,
       {
 	if (TopoDS::Edge(exp0.Current()).IsSame(TopoDS::Edge(exp1.Current())))
 	  {
-	    done = 1;
+	    done = true;
 	    double s0, s1;
 	    c = BRep_Tool::Curve(TopoDS::Edge(exp0.Current()), s0, s1);
 	  }
@@ -334,7 +460,66 @@ OCCRefinementSurfaces :: ~OCCRefinementSurfaces ()
 {
   ;
 }
+
+/*
+inline double Det3 (double a00, double a01, double a02,
+		    double a10, double a11, double a12,
+		    double a20, double a21, double a22)
+{
+  return a00*a11*a22 + a01*a12*a20 + a10*a21*a02 - a20*a11*a02 - a10*a01*a22 - a21*a12*a00;
+}
+
+bool ProjectToSurface (gp_Pnt & p, Handle(Geom_Surface) surface, double& u, double& v)
+{
+  gp_Pnt x = surface->Value (u,v);
+
+  if (p.SquareDistance(x) <= sqr(PROJECTION_TOLERANCE)) return true;
+
+  gp_Vec du, dv;
+
+  surface->D1(u,v,x,du,dv);
+
+  int count = 0;
+
+  gp_Pnt xold;
+  gp_Vec n;
+  double det, lambda, mu;
+
+  do {
+    count++;
+
+    n = du^dv;
+
+    det = Det3 (n.X(), du.X(), dv.X(),
+		n.Y(), du.Y(), dv.Y(),
+		n.Z(), du.Z(), dv.Z());
+
+    if (det < 1e-15) return false; 
+
+    lambda = Det3 (n.X(), p.X()-x.X(), dv.X(),
+	           n.Y(), p.Y()-x.Y(), dv.Y(),
+		   n.Z(), p.Z()-x.Z(), dv.Z())/det;
+
+    mu     = Det3 (n.X(), du.X(), p.X()-x.X(),
+		   n.Y(), du.Y(), p.Y()-x.Y(),
+		   n.Z(), du.Z(), p.Z()-x.Z())/det;
   
+    u += lambda;
+    v += mu;
+
+    xold = x;
+    surface->D1(u,v,x,du,dv);
+
+  } while (xold.SquareDistance(x) > sqr(PROJECTION_TOLERANCE) || count > 50);
+
+  if (count > 50) return false;
+
+  p = x;
+
+  return true;
+}
+*/
+
 void OCCRefinementSurfaces :: 
 PointBetween (const Point3d & p1, const Point3d & p2, double secpoint,
 	      int surfi, 
@@ -347,7 +532,16 @@ PointBetween (const Point3d & p1, const Point3d & p2, double secpoint,
 
   if (surfi > 0)
     {
-      geometry.Project (surfi, hnewp);
+      
+      double u = gi1.u+secpoint*(gi2.u-gi1.u);
+      double v = gi1.v+secpoint*(gi2.v-gi1.v);
+ 
+      if (!geometry.FastProject (surfi, hnewp, u, v))
+	{
+	  cout << "Fast projection to surface fails! Using OCC projection" << endl;
+          geometry.Project (surfi, hnewp);
+	}
+
       newgi.trignum = 1;
     }
   
@@ -370,6 +564,7 @@ PointBetween (const Point3d & p1, const Point3d & p2, double secpoint,
   pnt = proj.NearestPoint();
   hnewp = Point<3> (pnt.X(), pnt.Y(), pnt.Z());
   newp = hnewp;
+  newgi = ap1;
 };
 
 
@@ -379,6 +574,16 @@ void OCCRefinementSurfaces :: ProjectToSurface (Point<3> & p, int surfi)
     geometry.Project (surfi, p);
 };
 
+void OCCRefinementSurfaces :: ProjectToSurface (Point<3> & p, int surfi, PointGeomInfo & gi)
+{
+  if (surfi > 0)
+    if (!geometry.FastProject (surfi, p, gi.u, gi.v))
+      {
+	cout << "Fast projection to surface fails! Using OCC projection" << endl;
+        geometry.Project (surfi, p);
+      }
+};
+
 
 
 }
diff --git a/Netgen/libsrc/occ/occmeshsurf.hpp b/Netgen/libsrc/occ/occmeshsurf.hpp
index aa0399ac23..bda8a05a64 100644
--- a/Netgen/libsrc/occ/occmeshsurf.hpp
+++ b/Netgen/libsrc/occ/occmeshsurf.hpp
@@ -5,14 +5,24 @@
 
 #include "occgeom.hpp"
 
+#define PARAMETERSPACE -1
+#define PLANESPACE     1
+
 class OCCGeometry;
 
+class SingularMatrixException
+{};
+
+class UVBoundsException
+{};
+
 class OCCSurface
 {
 public:
   TopoDS_Face topods_face;
   Handle(Geom_Surface) occface;
   TopAbs_Orientation orient;
+  int projecttype;
 
 protected:
   Point<3> p1;
@@ -38,12 +48,22 @@ protected:
   Vec<2> psey;
   Mat<2,2> Amat, Amatinv;
 
+  // UV Bounds
+  double umin, umax, vmin, vmax;
+
 public:
-  OCCSurface (const TopoDS_Face & aface)
+  OCCSurface (const TopoDS_Face & aface, int aprojecttype)
   {
     topods_face = aface;
     occface = BRep_Tool::Surface(topods_face);
     orient = topods_face.Orientation();
+    projecttype = aprojecttype;
+    ShapeAnalysis::GetFaceUVBounds (topods_face, umin, umax, vmin, vmax);
+    umin -= fabs(umax-umin)/100.0;
+    vmin -= fabs(vmax-vmin)/100.0;
+    umax += fabs(umax-umin)/100.0;
+    vmax += fabs(vmax-vmin)/100.0;
+    // projecttype = PLANESPACE;
     /*
     TopExp_Explorer exp1;
     exp1.Init (topods_face, TopAbs_WIRE);
@@ -90,7 +110,11 @@ class Meshing2OCCSurfaces : public Meshing2
    
 public:
   ///
-  Meshing2OCCSurfaces (const TopoDS_Shape & asurf, const Box<3> & aboundingbox);
+  Meshing2OCCSurfaces (const TopoDS_Shape & asurf, const Box<3> & aboundingbox, int aprojecttype);
+
+  ///
+  int GetProjectionType ()
+  { return surface.projecttype; }
 
 protected:
   ///
@@ -164,6 +188,8 @@ public:
 			     Point3d & newp, EdgePointGeomInfo & newgi);
 
   virtual void ProjectToSurface (Point<3> & p, int surfi);
+
+  virtual void ProjectToSurface (Point<3> & p, int surfi, PointGeomInfo & gi);
 };
 
 
diff --git a/Netgen/libsrc/stlgeom/meshstlsurface.cpp b/Netgen/libsrc/stlgeom/meshstlsurface.cpp
index b912779237..4e50a908b4 100644
--- a/Netgen/libsrc/stlgeom/meshstlsurface.cpp
+++ b/Netgen/libsrc/stlgeom/meshstlsurface.cpp
@@ -24,9 +24,7 @@ static void STLFindEdges (STLGeometry & geom,
   // mark edge points:
   int ngp = geom.GetNP();
 
-  cout << "h1(0,0,0) = " << mesh.LocalHFunction().GetH (Point3d (0,0,0)) << endl;
   geom.RestrictLocalH(mesh, h);
-  cout << "h2(0,0,0) = " << mesh.LocalHFunction().GetH (Point3d (0,0,0)) << endl;
 
   PushStatusF("Mesh Lines");
 
@@ -324,6 +322,7 @@ int STLSurfaceMeshing (STLGeometry & geom,
 		  optmesh.ImproveMesh (mesh);
 		}
 
+	      mesh.Compress();
 	      mesh.FindOpenSegments();
 	      nopen = mesh.GetNOpenSegments();
 
@@ -707,6 +706,8 @@ void STLSurfaceOptimization (STLGeometry & geom,
 	if (multithread.terminate)
 	  break;
 
+	(*testout) << "optimize, before, step = " << mparam.optimize2d[j-1] << mesh.Point (3679) << endl;
+
 	mesh.CalcSurfacesOfNode();
 	switch (mparam.optimize2d[j-1])
 	  {
@@ -731,6 +732,7 @@ void STLSurfaceOptimization (STLGeometry & geom,
 	      break;
 	    }
 	  }
+	(*testout) << "optimize, after, step = " << mparam.optimize2d[j-1] << mesh.Point (3679) << endl;
       }
 
   geom.surfaceoptimized = 1;
@@ -1114,5 +1116,15 @@ void RefinementSTLGeometry :: ProjectToSurface (Point<3> & p, int surfi)
   cout << "RefinementSTLGeometry :: ProjectToSurface not implemented!" << endl;
 };
 
+
+void RefinementSTLGeometry :: ProjectToSurface (Point<3> & p, int surfi,
+						PointGeomInfo & gi)
+{
+  ((STLGeometry&)geom).SelectChartOfTriangle (gi.trignum);
+  gi.trignum = geom.Project (p);
+  //  if (!gi.trignum) 
+  //    cout << "projectSTL failed" << endl;
+};
+
  
 }
diff --git a/Netgen/libsrc/stlgeom/meshstlsurface.hpp b/Netgen/libsrc/stlgeom/meshstlsurface.hpp
index 99d2918fd1..f190d107a5 100644
--- a/Netgen/libsrc/stlgeom/meshstlsurface.hpp
+++ b/Netgen/libsrc/stlgeom/meshstlsurface.hpp
@@ -112,6 +112,7 @@ public:
 			     Point3d & newp, EdgePointGeomInfo & newgi);
 
   virtual void ProjectToSurface (Point<3> & p, int surfi);
+  virtual void ProjectToSurface (Point<3> & p, int surfi, PointGeomInfo & gi);
 };
 
 
diff --git a/Netgen/libsrc/stlgeom/stlgeom.cpp b/Netgen/libsrc/stlgeom/stlgeom.cpp
index d28ac37ff5..2d3a358cfa 100644
--- a/Netgen/libsrc/stlgeom/stlgeom.cpp
+++ b/Netgen/libsrc/stlgeom/stlgeom.cpp
@@ -29,9 +29,8 @@ void STLMeshing (STLGeometry & geom,
   geom.AddFaceEdges();
   geom.LinkEdges();
 
-  int i;
   mesh.ClearFaceDescriptors();
-  for (i = 1; i <= geom.GetNOFaces(); i++)
+  for (int i = 1; i <= geom.GetNOFaces(); i++)
     mesh.AddFaceDescriptor (FaceDescriptor (i, 1, 0, 0));
 }
 
diff --git a/Netgen/libsrc/stlgeom/stlgeom.hpp b/Netgen/libsrc/stlgeom/stlgeom.hpp
index 2ac1cd1e36..c9bfb1e535 100644
--- a/Netgen/libsrc/stlgeom/stlgeom.hpp
+++ b/Netgen/libsrc/stlgeom/stlgeom.hpp
@@ -54,7 +54,7 @@ namespace netgen
     void Store ();
     void Restore ();
 
-    void SetSize(int size) { };
+    void SetSize(int /* size */) { };
     void Clear() { };
     int Size() const { return geom.GetNTE(); }
     const STLTopEdge & Get(int i) const { return geom.GetTopEdge(i); }
diff --git a/Netgen/libsrc/stlgeom/stlgeomchart.cpp b/Netgen/libsrc/stlgeom/stlgeomchart.cpp
index 6e3fe8b2d3..c2f64f5f1f 100644
--- a/Netgen/libsrc/stlgeom/stlgeomchart.cpp
+++ b/Netgen/libsrc/stlgeom/stlgeomchart.cpp
@@ -17,7 +17,7 @@ int chartdebug = 0;
 
 
 
-void STLGeometry :: MakeAtlas(class Mesh & mesh)
+void STLGeometry :: MakeAtlas(Mesh & mesh)
 {
 
   double h, h2;
diff --git a/Netgen/libsrc/stlgeom/stlgeommesh.cpp b/Netgen/libsrc/stlgeom/stlgeommesh.cpp
index 37656495e7..00ce8fc001 100644
--- a/Netgen/libsrc/stlgeom/stlgeommesh.cpp
+++ b/Netgen/libsrc/stlgeom/stlgeommesh.cpp
@@ -14,7 +14,7 @@ namespace netgen
 {
 int EdgeUsed(int p1, int p2, ARRAY<INDEX_2>& edges, INDEX_2_HASHTABLE<int>& hashtab)
 {
-  if (p1 > p2) {Swap (p1,p2);}
+  if (p1 > p2) {swap (p1,p2);}
 
   if (hashtab.Used(INDEX_2(p1,p2))) 
     {return hashtab.Get(INDEX_2(p1,p2));}
@@ -75,7 +75,7 @@ Point<3> STLGeometry :: PointBetween(const Point<3> & p1, int t1,
       int ptn1 = GetTriangle(t1).PNum(i);
       int ptn2 = GetTriangle(t1).PNumMod(i+1);
 
-      if (ptn1 > ptn2) {Swap(ptn1,ptn2);}
+      if (ptn1 > ptn2) {swap(ptn1,ptn2);}
 
       Point3d pt1 = GetPoint(ptn1);
       Point3d pt2 = GetPoint(ptn2);
@@ -131,7 +131,7 @@ Point<3> STLGeometry :: PointBetween(const Point<3> & p1, int t1,
 		  int pnt1 = GetTriangle(tn).PNum(k);
 		  int pnt2 = GetTriangle(tn).PNumMod(k+1);
 		      
-		  if (pnt1 > pnt2) {Swap(pnt1,pnt2);}
+		  if (pnt1 > pnt2) {swap(pnt1,pnt2);}
 
 		  Point3d pt1 = GetPoint(pnt1);
 		  Point3d pt2 = GetPoint(pnt2);
@@ -1351,8 +1351,8 @@ int STLMeshingDummy (STLGeometry* stlgeometry, Mesh*& mesh,
       mesh -> SetLocalH (stlgeometry->GetBoundingBox().PMin() - Vec3d(10, 10, 10),
 			 stlgeometry->GetBoundingBox().PMax() + Vec3d(10, 10, 10),
 			 mparam.grading);
+      mesh -> LoadLocalMeshSize (mparam.meshsizefilename);
       
-      int i;
       success = 0;
   
       //mesh->DeleteMesh();
@@ -1451,7 +1451,7 @@ int STLMeshingDummy (STLGeometry* stlgeometry, Mesh*& mesh,
 	      mesh -> SetLocalH (stlgeometry->GetBoundingBox().PMin() - Vec3d(10, 10, 10),
 				 stlgeometry->GetBoundingBox().PMax() + Vec3d(10, 10, 10),
 				 mparam.grading);
-	      
+	      mesh -> LoadLocalMeshSize (mparam.meshsizefilename);	      
 	      mesh -> CalcLocalHFromSurfaceCurvature (stlparam.resthsurfmeshcurvfac);
 	      mparam.optimize2d = "cmsmSm";
 	      STLSurfaceOptimization (*stlgeometry, *mesh, mparam);
@@ -1502,6 +1502,7 @@ int STLMeshingDummy (STLGeometry* stlgeometry, Mesh*& mesh,
 	      mesh -> SetLocalH (stlgeometry->GetBoundingBox().PMin() - Vec3d(10, 10, 10),
 				 stlgeometry->GetBoundingBox().PMax() + Vec3d(10, 10, 10),
 				 mparam.grading);	  
+	      mesh -> LoadLocalMeshSize (mparam.meshsizefilename);
 	      mesh -> CalcLocalH ();
 	    }
 	  
@@ -1566,7 +1567,7 @@ int STLMeshingDummy (STLGeometry* stlgeometry, Mesh*& mesh,
 	    }
 
 
-	  OptimizeVolume (mparam, *mesh, NULL);
+	  OptimizeVolume (mparam, *mesh);
 	  
 #ifdef STAT_STREAM
 	  (*statout) << GetTime() << " & " << endl;
diff --git a/Netgen/libsrc/visualization/Makefile b/Netgen/libsrc/visualization/Makefile
index f8e80916e9..a6302092d9 100644
--- a/Netgen/libsrc/visualization/Makefile
+++ b/Netgen/libsrc/visualization/Makefile
@@ -1,7 +1,7 @@
 #
 # Makefile for visualization library
 #
-src = stlmeshing.cpp mvdraw.cpp vscsg.cpp vsmesh.cpp vssolution.cpp meshdoc.cpp
+src = stlmeshing.cpp mvdraw.cpp vscsg.cpp vsmesh.cpp vsocc.cpp vssolution.cpp meshdoc.cpp
 #
 lib = vis
 libpath = libsrc/visualization
diff --git a/Netgen/libsrc/visualization/mvdraw.cpp b/Netgen/libsrc/visualization/mvdraw.cpp
index c166e5aac3..265dc970b4 100644
--- a/Netgen/libsrc/visualization/mvdraw.cpp
+++ b/Netgen/libsrc/visualization/mvdraw.cpp
@@ -5,1536 +5,1331 @@
 #include <geometry2d.hpp>
 #include <stlgeom.hpp>
 
-#ifdef OCCGEOMETRY
-#include <occgeom.hpp>
-#include "TopoDS_Shape.hxx"
-#include "TopoDS_Vertex.hxx"
-#include "TopExp_Explorer.hxx"
-#include "BRep_Tool.hxx"
-#include "TopoDS.hxx"
-#include "gp_Pnt.hxx"
-#include "Geom_Curve.hxx"
-#include "Poly_Triangulation.hxx"
-#include "Poly_Array1OfTriangle.hxx"
-#include "TColgp_Array1OfPnt2d.hxx"
-#include "Poly_Triangle.hxx"
-#include "BRepMesh.hxx"
-#endif
-
-
 #include "incvis.hpp"
 
 
-#ifdef STEP
-#include <stepgeom.hpp>
-#endif
 
 namespace netgen
 {
 
 #include "mvdraw.hpp"
 
-Point3d VisualScene :: center;
-double VisualScene :: rad;
-double VisualScene :: backcolor;
-GLuint VisualScene :: fontbase = 0;
+  Point3d VisualScene :: center;
+  double VisualScene :: rad;
+  GLdouble VisualScene :: backcolor;
+  GLuint VisualScene :: fontbase = 0;
 
-// texture for color decoding
-GLubyte * VisualScene :: colortexture = NULL;
-GLuint VisualScene :: coltexname = 1;
-int VisualScene :: ntexcols = -1;
-  // bool VisualScene :: linear_colors = 1;
+  // texture for color decoding
+  GLubyte * VisualScene :: colortexture = NULL;
+  GLuint VisualScene :: coltexname = 1;
+  int VisualScene :: ntexcols = -1;
 
-float VisualScene :: lookatmat[16];
-float VisualScene :: transmat[16];
-float VisualScene :: rotmat[16];
-float VisualScene :: centermat[16];
-float VisualScene :: transformationmat[16];
 
+  float VisualScene :: lookatmat[16];
+  float VisualScene :: transmat[16];
+  float VisualScene :: rotmat[16];
+  float VisualScene :: centermat[16];
+  float VisualScene :: transformationmat[16];
 
 
-VisualizationParameters :: VisualizationParameters()
-{
-  lightamb = 0.3;
-  lightdiff = 0.7;
-  lightspec = 1;
-  shininess = 50;
-  transp = 0.3;
-  locviewer = 0;
-  showstltrias = 0;
-  centerpoint = 0;
-  usedispllists = 1;
-  strcpy (selectvisual, "cross");
-};
-VisualizationParameters vispar;
-
-
-
-double dist = 0;
-  // double dist = 6;
-// vorher: pnear = 2;
-double pnear = 0.1;
-double pfar = 10;
 
+  VisualizationParameters :: VisualizationParameters()
+  {
+    lightamb = 0.3;
+    lightdiff = 0.7;
+    lightspec = 1;
+    shininess = 50;
+    transp = 0.3;
+    locviewer = 0;
+    showstltrias = 0;
+    centerpoint = 0;
+    usedispllists = 1;
+    strcpy (selectvisual, "cross");
+
+    use_center_coords = false;
+  };
+  VisualizationParameters vispar;
 
 
-extern CSGeometry * geometry;
-extern STLGeometry * stlgeometry;
-extern SplineGeometry2d * geometry2d;
 
-#ifdef OCCGEOMETRY
-extern OCCGeometry * occgeometry;
-#endif
+  double dist = 0;
+  // double dist = 6;
+  // vorher: pnear = 2;
+  double pnear = 0.1;
+  double pfar = 10;
 
-#ifdef STEP
-using STEP_AP203::STEPGeometry;
-extern STEPGeometry * stepgeometry;
-#endif // STEP
-extern AutoPtr<Mesh> mesh;
-extern ARRAY<SpecialPoint> specpoints;
 
-//Tcl_Interp * hinterp;
 
+  extern STLGeometry * stlgeometry;
+  extern AutoPtr<SplineGeometry2d> geometry2d;
+  extern AutoPtr<Mesh> mesh;
+  extern ARRAY<SpecialPoint> specpoints;
 
-VisualScene :: VisualScene ()
-{
-  changeval = -1;
-  backcolor = 0;
 
+  VisualScene :: VisualScene ()
+  {
+    changeval = -1;
+    backcolor = 0;
+  }
 
 
-}
+  VisualScene :: ~VisualScene()
+  {
+    ;
+  }
 
-VisualScene :: ~VisualScene()
-{
-  ;
-}
 
+  void Render ()
+  {
+    multithread.redraw = 1;
+  }
 
-void Render ()
-{
-  multithread.redraw = 1;
-  //  while (multithread.redraw);
-      //   Tcl_Eval (hinterp, ".ndraw render");
-}
 
-void VisualScene :: BuildScene (int zoomall)
-{
-  center = Point3d (0,0,0);
-  rad = 1;
+  void VisualScene :: BuildScene (int zoomall)
+  {
+    center = Point3d (0,0,0);
+    rad = 1;
 
-  CalcTransformationMatrices();
+    CalcTransformationMatrices();
 
-  glEnable(GL_DEPTH_TEST);
-  glDisable (GL_DITHER);
+    glEnable(GL_DEPTH_TEST);
+    glDisable (GL_DITHER);
   
-  GLfloat ambvals[] = { 0.4f, 0.4f, 0.4f, 1.0f };
-  GLfloat diffvals[] = { 0.5f, 0.5f, 0.5f, 1.0f };
-  GLfloat specvals[] =  { 0.7f, 0.7f, 0.7f, 1.0f };
-  glLightfv(GL_LIGHT0, GL_AMBIENT, ambvals);
-  glLightfv(GL_LIGHT0, GL_DIFFUSE, diffvals);
-  glLightfv(GL_LIGHT0, GL_SPECULAR, specvals);
+    GLfloat ambvals[] = { 0.4f, 0.4f, 0.4f, 1.0f };
+    GLfloat diffvals[] = { 0.5f, 0.5f, 0.5f, 1.0f };
+    GLfloat specvals[] =  { 0.7f, 0.7f, 0.7f, 1.0f };
+    glLightfv(GL_LIGHT0, GL_AMBIENT, ambvals);
+    glLightfv(GL_LIGHT0, GL_DIFFUSE, diffvals);
+    glLightfv(GL_LIGHT0, GL_SPECULAR, specvals);
   
-  GLfloat light_position[] = { 1, 3, 3, 0 };
-  glLightfv(GL_LIGHT0, GL_POSITION, light_position);
+    GLfloat light_position[] = { 1, 3, 3, 0 };
+    glLightfv(GL_LIGHT0, GL_POSITION, light_position);
   
-  glLightModeli (GL_LIGHT_MODEL_TWO_SIDE, 0);
-  glEnable(GL_LIGHTING);
-  glEnable(GL_LIGHT0);
-}
+    glLightModeli (GL_LIGHT_MODEL_TWO_SIDE, 0);
+    glEnable(GL_LIGHTING);
+    glEnable(GL_LIGHT0);
+  }
 
 
-void VisualScene :: DrawScene ()
-{
-  if (changeval == -1)
-    BuildScene();
-  changeval = 0;
+  void VisualScene :: DrawScene ()
+  {
+    if (changeval == -1)
+      BuildScene();
+    changeval = 0;
 
-  glClearColor(backcolor, backcolor, backcolor, 1.0);
-  glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
+    glClearColor(backcolor, backcolor, backcolor, 1.0);
+    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
 
-  glEnable (GL_COLOR_MATERIAL);
-  glColor3f (1.0f, 1.0f, 1.0f);
-  glLineWidth (1.0f);
+    glEnable (GL_COLOR_MATERIAL);
+    glColor3f (1.0f, 1.0f, 1.0f);
+    glLineWidth (1.0f);
 
-  /*
-  glPushMatrix();
-  //  glLoadIdentity();
-  glMultMatrixf (transformationmat);
+    DrawCoordinateCross ();
+    DrawNetgenLogo ();
+    glFinish();  
+  }
+
+
+  void VisualScene :: CalcTransformationMatrices()
+  {
+
+    // prepare model view matrix
   
-  glBegin(GL_LINES);
-  glVertex3f (0.0f, 0.0f, 0.0f);
-  glVertex3f (1.0f, 0.0f, 0.0f);
-  glVertex3f (0.0f, 0.0f, 0.0f);
-  glVertex3f (0.0f, 1.0f, 0.0f);
-  glVertex3f (0.0f, 0.0f, 0.0f);
-  glVertex3f (0.0f, 0.0f, 1.0f);
-  glEnd ();
+    glPushMatrix();
+
+    glLoadIdentity();
+    gluLookAt (0, 0, 6, 0, 0, 0, 0, 1, 0);
+    glGetFloatv (GL_MODELVIEW_MATRIX, lookatmat);
+
+    glLoadIdentity();
+    glTranslatef(0.0f, 0.0f, -dist);
+    glGetFloatv (GL_MODELVIEW_MATRIX, transmat);
   
-  glPopMatrix();
-  */
+    glLoadIdentity();
+    glGetFloatv (GL_MODELVIEW_MATRIX, rotmat);
 
-  DrawCoordinateCross ();
-  DrawNetgenLogo ();
-  glFinish();  
-}
+    glScalef (1/rad, 1/rad, 1/rad);
+    glTranslated (-center.X(), -center.Y(), -center.Z());
+    glGetFloatv (GL_MODELVIEW_MATRIX, centermat);
 
+    glLoadIdentity();
+    glMultMatrixf (lookatmat);
+    glMultMatrixf (transmat);
+    glMultMatrixf (rotmat);
+    glMultMatrixf (centermat);
+    glGetFloatv (GL_MODELVIEW_MATRIX, transformationmat);
 
-void VisualScene :: CalcTransformationMatrices()
-{
+    glPopMatrix();
+  }
 
-  // prepare model view matrix
-  
-  glPushMatrix();
 
-  glLoadIdentity();
-  gluLookAt (0, 0, 6, 0, 0, 0, 0, 1, 0);
-  glGetFloatv (GL_MODELVIEW_MATRIX, lookatmat);
+  void VisualScene :: ArbitraryRotation (const ARRAY<double> & alpha, const ARRAY<Vec3d> & vec)
+  {
+    glPushMatrix();
+
+    glLoadIdentity();
 
-  glLoadIdentity();
-  glTranslatef(0.0f, 0.0f, -dist);
-  glGetFloatv (GL_MODELVIEW_MATRIX, transmat);
+    for(int i=0; i<alpha.Size() && i<vec.Size(); i++)
+      {
+	glRotatef(alpha[i], vec[i].X(), vec[i].Y(), vec[i].Z());
+      }
+
+    glGetFloatv (GL_MODELVIEW_MATRIX, rotmat);
+
+    glLoadIdentity();
+    glMultMatrixf (lookatmat);
+    glMultMatrixf (transmat);
+    glMultMatrixf (rotmat);
+    glMultMatrixf (centermat);
+    glGetFloatv (GL_MODELVIEW_MATRIX, transformationmat);
   
-  glLoadIdentity();
-  glGetFloatv (GL_MODELVIEW_MATRIX, rotmat);
+    glPopMatrix();
+  } 
 
-  glScalef (1/rad, 1/rad, 1/rad);
-  glTranslated (-center.X(), -center.Y(), -center.Z());
-  glGetFloatv (GL_MODELVIEW_MATRIX, centermat);
 
-  glLoadIdentity();
-  glMultMatrixf (lookatmat);
-  glMultMatrixf (transmat);
-  glMultMatrixf (rotmat);
-  glMultMatrixf (centermat);
-  glGetFloatv (GL_MODELVIEW_MATRIX, transformationmat);
 
-  glPopMatrix();
-}
+  void VisualScene :: ArbitraryRotation (const double alpha, const Vec3d & vec)
+  {
+    ARRAY<double> a(1); a[0] = alpha;
+    ARRAY<Vec3d> v(1); v[0] = vec;
 
-void VisualScene :: StandardRotation (const char * dir)
-{
-  glPushMatrix();
+    ArbitraryRotation(a,v);
+  } 
+
+  void VisualScene :: StandardRotation (const char * dir)
+  {
+    glPushMatrix();
 
-  glLoadIdentity();
+    glLoadIdentity();
   
-  if (strcmp (dir, "xy") == 0)
-    ;
-  else if (strcmp (dir, "yx") == 0)
-    glRotatef(180.0, 1.0f, 1.0f, 0.0f);    
-  else if (strcmp (dir, "xz") == 0)
-    glRotatef(-90.0, 1.0f, 0.0f, 0.0f);    
-  else if (strcmp (dir, "zx") == 0)
-    {
+    if (strcmp (dir, "xy") == 0)
+      ;
+    else if (strcmp (dir, "yx") == 0)
       glRotatef(180.0, 1.0f, 1.0f, 0.0f);    
+    else if (strcmp (dir, "xz") == 0)
       glRotatef(-90.0, 1.0f, 0.0f, 0.0f);    
-    }
-  else if (strcmp (dir, "yz") == 0)
-    {
-      glRotatef(-90.0, 0.0f, 0.0f, 1.0f);    
-      glRotatef(-90.0, 0.0f, 1.0f, 0.0f);    
-    }
-  else if (strcmp (dir, "zy") == 0)
-    glRotatef(90.0, 0.0f, 1.0f, 0.0f);    
-    
+    else if (strcmp (dir, "zx") == 0)
+      {
+	glRotatef(180.0, 1.0f, 1.0f, 0.0f);    
+	glRotatef(-90.0, 1.0f, 0.0f, 0.0f);    
+      }
+    else if (strcmp (dir, "yz") == 0)
+      {
+	glRotatef(-90.0, 0.0f, 0.0f, 1.0f);    
+	glRotatef(-90.0, 0.0f, 1.0f, 0.0f);    
+      }
+    else if (strcmp (dir, "zy") == 0)
+      glRotatef(90.0, 0.0f, 1.0f, 0.0f);    
+
 
-  glGetFloatv (GL_MODELVIEW_MATRIX, rotmat);
+    glGetFloatv (GL_MODELVIEW_MATRIX, rotmat);
 
-  glLoadIdentity();
-  glMultMatrixf (lookatmat);
-  glMultMatrixf (transmat);
-  glMultMatrixf (rotmat);
-  glMultMatrixf (centermat);
-  glGetFloatv (GL_MODELVIEW_MATRIX, transformationmat);
+    glLoadIdentity();
+    glMultMatrixf (lookatmat);
+    glMultMatrixf (transmat);
+    glMultMatrixf (rotmat);
+    glMultMatrixf (centermat);
+    glGetFloatv (GL_MODELVIEW_MATRIX, transformationmat);
   
-  glPopMatrix();
-}
+    glPopMatrix();
+  }
 
-void VisualScene :: MouseMove(int oldx, int oldy,
-			      int newx, int newy,
-			      char mode)
-{
-  int deltax = newx - oldx;
-  int deltay = newy - oldy;
+  void VisualScene :: MouseMove(int oldx, int oldy,
+				int newx, int newy,
+				char mode)
+  {
+    int deltax = newx - oldx;
+    int deltay = newy - oldy;
   
-  glPushMatrix();
-  glLoadIdentity ();
+    glPushMatrix();
+    glLoadIdentity ();
   
-  switch (mode)
-    {
-    case 'r':
-      {	
-	glRotatef(float(deltax)/2, 0.0f, 1.0f, 0.0f);
-	glRotatef(float(deltay)/2, 1.0f, 0.0f, 0.0f);
-	glMultMatrixf (rotmat);
-	glGetFloatv (GL_MODELVIEW_MATRIX, rotmat);
-	break;
-      }
-    case 'm':
+    switch (mode)
       {
-	GLdouble projmat[16], modelviewmat[16];
-	GLint viewport[4];
-	glGetDoublev (GL_PROJECTION_MATRIX, projmat);
-	glGetDoublev (GL_MODELVIEW_MATRIX, modelviewmat);
-	glGetIntegerv (GL_VIEWPORT, viewport);
+      case 'r':
+	{	
+	  glRotatef(float(deltax)/2, 0.0f, 1.0f, 0.0f);
+	  glRotatef(float(deltay)/2, 1.0f, 0.0f, 0.0f);
+	  glMultMatrixf (rotmat);
+	  glGetFloatv (GL_MODELVIEW_MATRIX, rotmat);
+	  break;
+	}
+      case 'm':
+	{
+	  GLdouble projmat[16], modelviewmat[16];
+	  GLint viewport[4];
+	  glGetDoublev (GL_PROJECTION_MATRIX, projmat);
+	  glGetDoublev (GL_MODELVIEW_MATRIX, modelviewmat);
+	  glGetIntegerv (GL_VIEWPORT, viewport);
 	
-	// vorher pvz1/2 = 0
-	GLdouble pvx1 = 0, pvy1 = 0, pvz1 = 0.95;
-	GLdouble pvx2 = deltax, pvy2 = -deltay, pvz2 = 0.95;
+	  // vorher pvz1/2 = 0
+	  GLdouble pvx1 = 0, pvy1 = 0, pvz1 = 0.95;
+	  GLdouble pvx2 = deltax, pvy2 = -deltay, pvz2 = 0.95;
 
-	GLdouble px1, py1, pz1;
-	GLdouble px2, py2, pz2;
+	  GLdouble px1, py1, pz1;
+	  GLdouble px2, py2, pz2;
 	
-	gluUnProject (pvx1, pvy1, pvz1, 
-		      modelviewmat, projmat, viewport,
-		      &px1, &py1, &pz1);
-	gluUnProject (pvx2, pvy2, pvz2, 
-		      modelviewmat, projmat, viewport,
-		      &px2, &py2, &pz2);
-	/*
-	gluUnProject (oldx, oldy, 1, 
-		      modelviewmat, projmat, viewport,
-		      &px1, &py1, &pz1);
-	gluUnProject (newx, newy, 1, 
-		      modelviewmat, projmat, viewport,
-		      &px2, &py2, &pz2);
-	*/
-
-	/*	
-	cout << "pv1 = " << pvx1 << ", " << pvy1 << ", " << pvz1 << endl;
-	cout << "p1 = " << px1 << ", " << py1 << ", " << pz1 << endl;
-	*/
-
-	glTranslated (px2-px1, py2-py1, pz2-pz1);
+	  gluUnProject (pvx1, pvy1, pvz1, 
+			modelviewmat, projmat, viewport,
+			&px1, &py1, &pz1);
+	  gluUnProject (pvx2, pvy2, pvz2, 
+			modelviewmat, projmat, viewport,
+			&px2, &py2, &pz2);
+	  /*
+	    gluUnProject (oldx, oldy, 1, 
+	    modelviewmat, projmat, viewport,
+	    &px1, &py1, &pz1);
+	    gluUnProject (newx, newy, 1, 
+	    modelviewmat, projmat, viewport,
+	    &px2, &py2, &pz2);
+	  */
+
+	  /*	
+	    cout << "pv1 = " << pvx1 << ", " << pvy1 << ", " << pvz1 << endl;
+	    cout << "p1 = " << px1 << ", " << py1 << ", " << pz1 << endl;
+	  */
+
+	  glTranslated (px2-px1, py2-py1, pz2-pz1);
 	
-	glMultMatrixf (transmat);
-	glGetFloatv (GL_MODELVIEW_MATRIX, transmat);
-	break;
-      }
-    case 'z':
-      {
-	// glTranslatef(0.0f, 0.0f, -dist);
-	glScaled (exp (float (-deltay)/100), 
-		  exp (float (-deltay)/100), 
-		  exp (float (-deltay)/100));
-	// glTranslatef(0.0f, 0.0f, dist);
-	glMultMatrixf (transmat);
-	glGetFloatv (GL_MODELVIEW_MATRIX, transmat);
-	break;
+	  glMultMatrixf (transmat);
+	  glGetFloatv (GL_MODELVIEW_MATRIX, transmat);
+	  break;
+	}
+      case 'z':
+	{
+	  // glTranslatef(0.0f, 0.0f, -dist);
+	  glScaled (exp (float (-deltay)/100), 
+		    exp (float (-deltay)/100), 
+		    exp (float (-deltay)/100));
+	  // glTranslatef(0.0f, 0.0f, dist);
+	  glMultMatrixf (transmat);
+	  glGetFloatv (GL_MODELVIEW_MATRIX, transmat);
+	  break;
+	}
       }
-    }
 
-  glLoadIdentity();
-  glMultMatrixf (lookatmat);
-  glMultMatrixf (transmat);
-  glMultMatrixf (rotmat);
-  glMultMatrixf (centermat);
-  glGetFloatv (GL_MODELVIEW_MATRIX, transformationmat);
+    glLoadIdentity();
+    glMultMatrixf (lookatmat);
+    glMultMatrixf (transmat);
+    glMultMatrixf (rotmat);
+    glMultMatrixf (centermat);
+    glGetFloatv (GL_MODELVIEW_MATRIX, transformationmat);
   
-  glPopMatrix();
-}
+    glPopMatrix();
+  }
 
 
-void VisualScene :: LookAt (const Point<3> & cam, const Point<3> & obj,
-			    const Point<3> & camup)
-{
-  glPushMatrix();
-  glLoadIdentity ();
-  gluLookAt (cam(0), cam(1), cam(2), 
-	     obj(0), obj(1), obj(2),
-	     camup(0), camup(1), camup(2));
-  glMultMatrixf (centermat);
-  glGetFloatv (GL_MODELVIEW_MATRIX, transformationmat);
-  glPopMatrix();
-}
+  void VisualScene :: LookAt (const Point<3> & cam, const Point<3> & obj,
+			      const Point<3> & camup)
+  {
+    glPushMatrix();
+    glLoadIdentity ();
+    gluLookAt (cam(0), cam(1), cam(2), 
+	       obj(0), obj(1), obj(2),
+	       camup(0), camup(1), camup(2));
+    glMultMatrixf (centermat);
+    glGetFloatv (GL_MODELVIEW_MATRIX, transformationmat);
+    glPopMatrix();
+  }
 
   
-void VisualScene :: SetClippingPlane ()
-{
-  if (vispar.clipenable)
-    {
-      Vec3d n = vispar.clipnormal;
-      n /= (n.Length()+1e-10);
-      clipplane[0] = n.X();
-      clipplane[1] = n.Y();
-      clipplane[2] = n.Z();
-      clipplane[3] = -(Vec3d(center) * n) + rad * vispar.clipdist;
-
-      glClipPlane(GL_CLIP_PLANE0, clipplane);
-      glEnable(GL_CLIP_PLANE0);
-    }
-  else
-    glDisable (GL_CLIP_PLANE0);
-}
+  void VisualScene :: SetClippingPlane ()
+  {
+    if (vispar.clipenable)
+      {
+	Vec3d n = vispar.clipnormal;
+	n /= (n.Length()+1e-10);
+	clipplane[0] = n.X();
+	clipplane[1] = n.Y();
+	clipplane[2] = n.Z();
+	clipplane[3] = -(Vec3d(center) * n) + rad * vispar.clipdist;
+
+	glClipPlane(GL_CLIP_PLANE0, clipplane);
+	glEnable(GL_CLIP_PLANE0);
+      }
+    else
+      glDisable (GL_CLIP_PLANE0);
+  }
 
 
 
 
-void VisualScene :: MouseDblClick (int px, int py)
-{
-  ;
-}
+  void VisualScene :: MouseDblClick (int /* px */, int /* py */)
+  {
+    ;
+  }
 
 
 
-void VisualScene :: SetLight()
-{
-  GLfloat vals[3];
-  double lightamb = vispar.lightamb;
-  vals[0] = vals[1] = vals[2] = lightamb;
-  glLightfv(GL_LIGHT0, GL_AMBIENT, vals);
+  void VisualScene :: SetLight()
+  {
+    GLfloat vals[3];
+    double lightamb = vispar.lightamb;
+    vals[0] = vals[1] = vals[2] = lightamb;
+    glLightfv(GL_LIGHT0, GL_AMBIENT, vals);
 
-  double lightdiff = vispar.lightdiff;
-  vals[0] = vals[1] = vals[2] = lightdiff;
-  glLightfv(GL_LIGHT0, GL_DIFFUSE, vals);
+    double lightdiff = vispar.lightdiff;
+    vals[0] = vals[1] = vals[2] = lightdiff;
+    glLightfv(GL_LIGHT0, GL_DIFFUSE, vals);
 
-  double lightspec = vispar.lightspec;
-  vals[0] = vals[1] = vals[2] = lightspec;
-  glLightfv(GL_LIGHT0, GL_SPECULAR, vals);
+    double lightspec = vispar.lightspec;
+    vals[0] = vals[1] = vals[2] = lightspec;
+    glLightfv(GL_LIGHT0, GL_SPECULAR, vals);
 
-  glMaterialf (GL_FRONT_AND_BACK, GL_SHININESS, vispar.shininess);
-  glLightModeli (GL_LIGHT_MODEL_LOCAL_VIEWER, vispar.locviewer);
+    glMaterialf (GL_FRONT_AND_BACK, GL_SHININESS, vispar.shininess);
+    glLightModeli (GL_LIGHT_MODEL_LOCAL_VIEWER, vispar.locviewer);
 
-  float mat_spec_col[] = { 1, 1, 1, 1 };
-  glMaterialfv (GL_FRONT_AND_BACK, GL_SPECULAR, mat_spec_col);
+    float mat_spec_col[] = { 1, 1, 1, 1 };
+    glMaterialfv (GL_FRONT_AND_BACK, GL_SPECULAR, mat_spec_col);
 
-  glEnable (GL_LIGHTING);
-  glEnable (GL_LIGHT0);
-}
+    glEnable (GL_LIGHTING);
+    glEnable (GL_LIGHT0);
+  }
 
 
 
 
-void VisualScene :: SetOpenGlColor(double h, double hmin, double hmax,
-				   int logscale)
-{
-  double value;
+  void VisualScene :: SetOpenGlColor(double h, double hmin, double hmax,
+				     int logscale)
+  {
+    double value;
 
-  if (!logscale)
-    value = (h - hmin) / (hmax - hmin);
-  else
-    {
-      if (hmax <= 0) hmax = 1;
-      if (hmin <= 0) hmin = 1e-4 * hmax;
-      value = (log(fabs(h)) - log(hmin)) / (log(hmax) - log(hmin));
-    }
+    if (!logscale)
+      value = (h - hmin) / (hmax - hmin);
+    else
+      {
+	if (hmax <= 0) hmax = 1;
+	if (hmin <= 0) hmin = 1e-4 * hmax;
+	value = (log(fabs(h)) - log(hmin)) / (log(hmax) - log(hmin));
+      }
 
-  if (!invcolor)
-    value = 1 - value;
+    if (!invcolor)
+      value = 1 - value;
 
-  glTexCoord1f ( 0.999 * value + 0.001);
+    glTexCoord1f ( 0.999 * value + 0.001);
+    // glTexCoord1f ( value ); 
 
-  if (value > 1) value = 1;
-  if (value < 0) value = 0;
+    if (value > 1) value = 1;
+    if (value < 0) value = 0;
 
-  value *= 4;
+    value *= 4;
 
-  const double colp[][3] =
-  {
-    { 1, 0, 0 },
-    { 1, 1, 0 },
-    { 0, 1, 0 },
-    { 0, 1, 1 },
-    { 0, 0, 1 },
-    { 1, 0, 1 },
-    { 1, 0, 0 },
-  };
+    static const double colp[][3] =
+      {
+	{ 1, 0, 0 },
+	{ 1, 1, 0 },
+	{ 0, 1, 0 },
+	{ 0, 1, 1 },
+	{ 0, 0, 1 },
+	//	{ 1, 0, 1 },
+	//	{ 1, 0, 0 },
+      };
   
-  int i = int(value);
-  double r = value - i;
+    int i = int(value);
+    double r = value - i;
 
-  GLdouble col[3];
-  int j;
-  for (j = 0; j < 3; j++)
-    col[j] = (1-r) * colp[i][j] + r * colp[i+1][j];
+    GLdouble col[3];
+    for (int j = 0; j < 3; j++)
+      col[j] = (1-r) * colp[i][j] + r * colp[i+1][j];
   
-  glColor3d (col[0], col[1], col[2]);
-}
-
-
-void VisualScene :: CreateTexture (int ncols, int linear)
-{
-  int i;
+    glColor3d (col[0], col[1], col[2]);
+  }
 
-  if (ncols < 2) ncols = 2;
-  ncols = 8;
 
-  if (ntexcols != ncols)
-    {
-      if (colortexture) 
-	{
-	  glDeleteTextures (1, &coltexname);
-	  delete colortexture;
-	}
-      
-      ntexcols = ncols;
-      
-      colortexture = new GLubyte[4*ncols+4];
+  /*
+  void VisualScene :: CreateTexture (int ncols, int linear, int typ)
+  {
 
-      const double colp[][3] =
+    static const double colp[][3] =
       {
 	{ 1, 0, 0 },
 	{ 1, 1, 0 },
 	{ 0, 1, 0 },
 	{ 0, 1, 1 },
 	{ 0, 0, 1 },
-	{ 1, 0, 1 },
-	{ 1, 0, 0 },
       };
   
-      for (i = 0; i < ncols; i++)
-	{
-	  double value = 4.0 * i / (ncols-1);
-
-	  int iv = int(value);
-	  double r = value - iv;
 
-	  GLdouble col[3];
-	  int j;
-	  for (j = 0; j < 3; j++)
-	    col[j] = (1-r) * colp[iv][j] + r * colp[iv+1][j];
-	  
-	  glColor3d (col[0], col[1], col[2]);
 
-	  colortexture[4*i] = GLubyte (255 * col[0]);
-	  colortexture[4*i+1] = GLubyte (255 * col[1]);
-	  colortexture[4*i+2] = GLubyte (255 * col[2]);
-	  colortexture[4*i+3] = GLubyte(255);
-	}
+    if (ntexcols != 1024)
+      {
+	ntexcols = 1024;
+	
+	// glPixelStorei (GL_UNPACK_ALIGNMENT, 1);
+	glGenTextures (1, &coltexname);
+	glBindTexture (GL_TEXTURE_1D, coltexname);
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_WRAP_S, GL_CLAMP);
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MIN_FILTER, GL_NEAREST_MIPMAP_NEAREST);
+	
+	for (int level = 0; level <= 11; level++)
+	  {
+	    ncols = 2048 >> level;
+	    cout << "ncols = " << ncols << endl;
 
-      //      glPixelStorei (GL_UNPACK_ALIGNMENT, 1);
+	    colortexture = new GLubyte[4*ncols+12];
 
-      
-      glGenTextures (1, &coltexname);
+	    for (int i = 0; i < ncols; i++)
+	      {
+		double value = 4.0 * i / (ncols-1);
+		
+		int iv = int(value);
+		double r = value - iv;
+		
+		GLdouble col[3];
+		for (int j = 0; j < 3; j++)
+		  col[j] = (1-r) * colp[iv][j] + r * colp[iv+1][j];
+		
+		colortexture[4*i] = GLubyte (255 * col[0]);
+		colortexture[4*i+1] = GLubyte (255 * col[1]);
+		colortexture[4*i+2] = GLubyte (255 * col[2]);
+		colortexture[4*i+3] = GLubyte(255);
+		
+		if (ncols > 20)
+		  if ( i % (ncols / 10) == 0)
+		    {
+		      colortexture[4*i] = GLubyte (0);
+		      colortexture[4*i+1] = GLubyte (0);
+		      colortexture[4*i+2] = GLubyte (0);
+		      colortexture[4*i+4] = GLubyte (0);
+		      colortexture[4*i+5] = GLubyte (0);
+		      colortexture[4*i+6] = GLubyte (0);
+		    }
+	      }
+	    
+	    glTexImage1D (GL_TEXTURE_1D, level, 4, ncols, 0, GL_RGBA, GL_UNSIGNED_BYTE, colortexture);
+	  }
+      }
+    
+    glTexEnvf (GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, typ);  // DECAL or MODULATE
+    glBindTexture (GL_TEXTURE_1D, coltexname);
+  }
+  */
 
 
-      glBindTexture (GL_TEXTURE_1D, coltexname);
+  void VisualScene :: CreateTexture (int ncols, int linear, int typ)
+  {
+    if (ncols < 2) ncols = 2;
 
-      glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_WRAP_S, GL_CLAMP);
-      glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
-      glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
+    if (linear) ncols = 32;
+    else   ncols = 8;
 
-      glTexImage1D (GL_TEXTURE_1D, 0, 4, ncols, 0, GL_RGBA, GL_UNSIGNED_BYTE, colortexture);
-      glTexEnvf (GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_DECAL);
- 
-    }
+    if (ntexcols != ncols) 
+      {
+	if (colortexture) 
+	  {
+	    glDeleteTextures (1, &coltexname);
+	    delete colortexture;
+	  }
+      
+	ntexcols = ncols;
+      
+	colortexture = new GLubyte[4*ncols+12];
 
+	const double colp[][3] =
+	  {
+	    { 1, 0, 0 },
+	    { 1, 1, 0 },
+	    { 0, 1, 0 },
+	    { 0, 1, 1 },
+	    { 0, 0, 1 },
+	  };
+  
+	for (int i = 0; i < ncols; i++)
+	  {
+	    double value = 4.0 * i / (ncols-1);
 
-  glBindTexture (GL_TEXTURE_1D, coltexname);
-  if (linear)
-    {
-      glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
-      glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
-    }
-  else
-    {
-      glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
-      glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
-    }
-}
+	    int iv = int(value);
+	    double r = value - iv;
 
+	    GLdouble col[3];
+	    for (int j = 0; j < 3; j++)
+	      col[j] = (1-r) * colp[iv][j] + r * colp[iv+1][j];
+	  
+	    colortexture[4*i] = GLubyte (255 * col[0]);
+	    colortexture[4*i+1] = GLubyte (255 * col[1]);
+	    colortexture[4*i+2] = GLubyte (255 * col[2]);
+	    colortexture[4*i+3] = GLubyte(255);
+	  }
 
+	//      glPixelStorei (GL_UNPACK_ALIGNMENT, 1);
 
+	glGenTextures (1, &coltexname);
+	glBindTexture (GL_TEXTURE_1D, coltexname);
 
-void VisualScene :: DrawColorBar (double minval, double maxval,
-				  int logscale)
-{
-  int i;
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_WRAP_S, GL_CLAMP);
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
 
-  if (!vispar.drawcolorbar) return;
+	glTexImage1D (GL_TEXTURE_1D, 0, 4, ncols, 0, GL_RGBA, GL_UNSIGNED_BYTE, colortexture);
+      }
 
-  CreateTexture (8, 1);
+    glTexEnvf (GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, typ);  // DECAL or MODULATE
 
-  if (logscale && maxval <= 0) maxval = 1;
-  if (logscale && minval <= 0) minval = 1e-4 * maxval;
+    glBindTexture (GL_TEXTURE_1D, coltexname);
+    if (linear)
+      {
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
+      }
+    else
+      {
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
+      }
+  }
 
 
-  double minx = -1;
-  double maxx = 1;
-  double miny = 0.75;
-  double maxy = 0.8;
-      
-  double x;
 
-  glEnable (GL_COLOR_MATERIAL);
-  glEnable (GL_TEXTURE_1D);
-  glNormal3d (0, 0, 1);
 
-  glBegin (GL_LINES);
-  for (x = minx; x <= maxx; x += (maxx - minx) / 1000)
-    {
-      SetOpenGlColor (x, minx, maxx);
-      glVertex3d (x, miny,-5);
-      glVertex3d (x, maxy,-5);
-    }
-  glEnd();
 
-  glDisable (GL_TEXTURE_1D);
+  /*
+  void VisualScene :: CreateTexture (int ncols, int linear, int typ)
+  {
+    if (ncols < 2) ncols = 2;
 
-  glEnable (GL_COLOR_MATERIAL);
-  GLfloat textcol[3] = { 1 - backcolor,
-			 1 - backcolor,
-			 1 - backcolor };
-  glColor3fv (textcol);
-  glPushAttrib (GL_LIST_BIT);
-  glListBase (fontbase);
+    if (linear) ncols = 32;
+    else   ncols = 8;
 
-  char buf[20];
-  for (i = 0; i <= 4; i++)
-    {
-      x = minx + i * (maxx-minx) / 4;
-      glRasterPos3d (x, 0.7,-5);
+    if (ntexcols != ncols) 
+      {
+	if (colortexture) 
+	  {
+	    glDeleteTextures (1, &coltexname);
+	    delete colortexture;
+	  }
       
-      double val;
-      if (logscale)
-	val = minval * pow (maxval / minval, i / 4.0);
-      else
-	val = minval + i * (maxval-minval) / 4;
-
-      sprintf (buf, "%8.3e", val);
-      glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
-    }
-
-  glPopAttrib ();
-}
-
-
-void VisualScene :: DrawCoordinateCross ()
-{
-  if (!vispar.drawcoordinatecross) return;
-
-  glMatrixMode (GL_PROJECTION); 
-  glPushMatrix();
-  glLoadIdentity();
-
-  glMatrixMode (GL_MODELVIEW); 
-  glPushMatrix();
-  glLoadIdentity();
-
-  GLint viewport[4];
-  glGetIntegerv (GL_VIEWPORT, viewport);
-
-  glTranslatef (-1, -1, 0.0);
-  glScalef (40.0 / viewport[2], 40.0 / viewport[3], 1);
-  glTranslatef (2.0, 2.0, 0.0);
-  glMultMatrixf (rotmat);
-
-  glEnable (GL_COLOR_MATERIAL);
-  glDisable (GL_LIGHTING);
-
-  GLfloat textcol[3] = { 1 - backcolor,
-			 1 - backcolor,
-			 1 - backcolor };
-  glColor3fv (textcol);
-
-  glLineWidth (1.0f);
-
-  float len = 1;
-  glBegin(GL_LINES);
-  glVertex3f (0.0f, 0.0f, 0.0f);
-  glVertex3f (len, 0.0f, 0.0f);
-  glVertex3f (0.0f, 0.0f, 0.0f);
-  glVertex3f (0.0f, len, 0.0f);
-  glVertex3f (0.0f, 0.0f, 0.0f);
-  glVertex3f (0.0f, 0.0f, len);
-  glEnd ();
-
-
-  glPushAttrib (GL_LIST_BIT);
-  glListBase (fontbase);
-
-  char buf[20];
-
-  glRasterPos3d (len, 0.0f, 0.0f);
-  sprintf (buf, "x");
-  glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
-  glRasterPos3d (0.0f, len, 0.0f);
-  sprintf (buf, "y");
-  glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
-  glRasterPos3d (0.0f, 0.0f, len);
-  sprintf (buf, "z");
-  glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
-
-  glPopAttrib ();
-  glEnable (GL_LIGHTING);
-
-  glMatrixMode (GL_PROJECTION); 
-  glPopMatrix();
-  glMatrixMode (GL_MODELVIEW); 
-  glPopMatrix();
-}
-
-
-void VisualScene :: DrawNetgenLogo ()
-{
-  if (!vispar.drawnetgenlogo) return;
-
-  glMatrixMode (GL_PROJECTION); 
-  glPushMatrix();
-  glLoadIdentity();
-
-  glMatrixMode (GL_MODELVIEW); 
-  glPushMatrix();
-  glLoadIdentity();
-
-  GLint viewport[4];
-  glGetIntegerv (GL_VIEWPORT, viewport);
-
+	ntexcols = ncols;
+      
+	colortexture = new GLubyte[4*ncols+12];
 
+	const double colp[][3] =
+	  {
+	    { 1, 0, 0 },
+	    { 1, 1, 0 },
+	    { 0, 1, 0 },
+	    { 0, 1, 1 },
+	    { 0, 0, 1 },
+	  };
+  
+	for (int i = 0; i < ncols; i++)
+	  {
+	    double value = 4.0 * i / (ncols-1);
 
-  glTranslatef (1, -1, 0.0);
-  glScalef (40.0 / viewport[2], 40.0 / viewport[3], 1);
-  glTranslatef (-6.0, 2.0, 0.0);
-  //  glMultMatrixf (rotmat);
+	    int iv = int(value);
+	    double r = value - iv;
 
-  glEnable (GL_COLOR_MATERIAL);
-  GLfloat textcol[3] = { 1 - backcolor,
-			 1 - backcolor,
-			 1 - backcolor };
-  glColor3fv (textcol);
+	    GLdouble col[3];
+	    for (int j = 0; j < 3; j++)
+	      col[j] = (1-r) * colp[iv][j] + r * colp[iv+1][j];
+	  
+	    colortexture[4*i+4] = GLubyte (255 * col[0]);
+	    colortexture[4*i+5] = GLubyte (255 * col[1]);
+	    colortexture[4*i+6] = GLubyte (255 * col[2]);
+	    colortexture[4*i+7] = GLubyte(255);
+	  }
+	for (int j = 0; j < 4; j++)
+	  {
+	    colortexture[j] = colortexture[4+j];
+	    colortexture[ncols*4+4+j] = colortexture[ncols*4+j];
+	  }
 
-  glLineWidth (1.0f);
+	//      glPixelStorei (GL_UNPACK_ALIGNMENT, 1);
 
-  glPushAttrib (GL_LIST_BIT);
-  glListBase (fontbase);
+	glGenTextures (1, &coltexname);
+	glBindTexture (GL_TEXTURE_1D, coltexname);
 
-  char buf[20];
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_WRAP_S, GL_CLAMP);
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
 
-  glRasterPos3d (0.0f, 0.0f, 0.0f);
-  sprintf (buf, "Netgen 4.3.1");
-  glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
-  glPopAttrib ();
+	glTexImage1D (GL_TEXTURE_1D, 0, 4, ncols, 0, GL_RGBA, GL_UNSIGNED_BYTE, colortexture+4);
+	int bcol[] = { 0, 0, -1, -1 };
+	glTexParameteriv (GL_TEXTURE_1D, GL_TEXTURE_BORDER_COLOR, bcol);
+      }
 
-  glMatrixMode (GL_PROJECTION); 
-  glPopMatrix();
-  glMatrixMode (GL_MODELVIEW); 
-  glPopMatrix();
-}
+    glTexEnvf (GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, typ);  // DECAL or MODULATE
 
+    glBindTexture (GL_TEXTURE_1D, coltexname);
+    if (linear)
+      {
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
+      }
+    else
+      {
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
+	glTexParameteri (GL_TEXTURE_1D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
+      }
+  }
+  */
 
 
 
 
 
+  void VisualScene :: DrawColorBar (double minval, double maxval, int logscale, bool linear)
+  {
+    if (!vispar.drawcolorbar) return;
 
+    CreateTexture (8, linear, GL_DECAL);
 
-VisualSceneGeometry2d :: VisualSceneGeometry2d ()
-  : VisualScene()
-{
-  ;
-}
+    if (logscale && maxval <= 0) maxval = 1;
+    if (logscale && minval <= 0) minval = 1e-4 * maxval;
 
-VisualSceneGeometry2d :: ~VisualSceneGeometry2d ()
-{
-  ;
-}
+    double minx = -1;
+    double maxx = 1;
+    double miny = 0.75;
+    double maxy = 0.8;
 
+    glEnable (GL_COLOR_MATERIAL);
+    glEnable (GL_TEXTURE_1D);
+    glNormal3d (0, 0, 1);
+    glPolygonMode (GL_FRONT_AND_BACK, GL_FILL);
+    
+    glDisable (GL_DEPTH_TEST);
+    glBegin (GL_QUAD_STRIP);
 
+    for (double x = minx; x <= maxx; x += (maxx - minx) / 50)
+      {
+	SetOpenGlColor (x, minx, maxx);
+	// glTexCoord1f ( 0.999 * (x-minx) / (maxx-minx) + 0.001);
 
+	glVertex3d (x, miny, -5);
+	glVertex3d (x, maxy, -5);
+      }
+    glEnd();
 
+    glDisable (GL_TEXTURE_1D);
 
 
-void VisualSceneGeometry2d :: DrawScene ()
-{
-  int i, j;
+    glEnable (GL_COLOR_MATERIAL);
+    GLfloat textcol[3] = { 1 - backcolor, 1 - backcolor, 1 - backcolor };
+    glColor3fv (textcol);
 
-  if (changeval != geometry2d->GetSplines().Size())
-    BuildScene();
-  changeval = geometry2d->GetSplines().Size();
+    glPushAttrib (GL_LIST_BIT);
+    glListBase (fontbase);
 
-  
-  glClearColor(backcolor, backcolor, backcolor, 1.0);
-  glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
+    char buf[20];
+    for (int i = 0; i <= 4; i++)
+      {
+	double x = minx + i * (maxx-minx) / 4;
+	glRasterPos3d (x, 0.7,-5);
+      
+	double val;
+	if (logscale)
+	  val = minval * pow (maxval / minval, i / 4.0);
+	else
+	  val = minval + i * (maxval-minval) / 4;
+
+	sprintf (buf, "%8.3e", val);
+	glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
+      }
 
-  SetLight();
+    glPopAttrib ();
+    glEnable (GL_DEPTH_TEST);
+  }
 
-  //  glEnable (GL_LIGHT0);
-  glDisable (GL_LIGHTING);
-  glPushMatrix();
-  glMultMatrixf (transformationmat);
 
-  //  SetClippingPlane ();
+  void VisualScene :: DrawCoordinateCross ()
+  {
+    if (!vispar.drawcoordinatecross) return;
 
-  glShadeModel (GL_SMOOTH);
-  glEnable (GL_COLOR_MATERIAL);
-  glPolygonMode (GL_FRONT_AND_BACK, GL_LINE);
-  
-  //  float mat_col[] = { 0, 0, 1, 1 };
-  //  glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col);
-  glColor3f (0, 0, 1);
-  
+    glDisable (GL_DEPTH_TEST);
+    glMatrixMode (GL_PROJECTION); 
+    glPushMatrix();
+    glLoadIdentity();
 
-  ARRAY<Point<2> > points, otherpoints;
+    glMatrixMode (GL_MODELVIEW); 
+    glPushMatrix();
+    glLoadIdentity();
 
-  for (i = 1; i <= geometry2d->GetSplines().Size(); i++)
-    {
-      geometry2d->GetSplines().Get(i)->GetPoints (20, points);
-      
-      glBegin (GL_LINE_STRIP);
-      for (j = 0; j < points.Size(); j++)
-	glVertex3f (points[j](0), points[j](1), 0);
-      glEnd(); 
-    }
+    GLint viewport[4];
+    glGetIntegerv (GL_VIEWPORT, viewport);
 
-  glColor3f (1, 0, 0);
+    glTranslatef (-1, -1, 0.0);
+    glScalef (40.0 / viewport[2], 40.0 / viewport[3], 1);
+    glTranslatef (2.0, 2.0, 0.0);
+    glMultMatrixf (rotmat);
 
-  for (i = 1; i <= geometry2d->GetSplines().Size(); i++)
-    {
-      int other = geometry2d->GetSplines().Get(i)->copyfrom;
-      if (other != -1)
-	{
-	  geometry2d->GetSplines().Get(i)->GetPoints (6, points);
-	  geometry2d->GetSplines().Get(other)->GetPoints (6, otherpoints);
-	  glBegin (GL_LINES);
-	  for (j = 1; j < 5; j++)
-	    {
-	      glVertex3f (points[j](0), points[j](1), 0);
-	      glVertex3f (otherpoints[j](0), otherpoints[j](1), 0);
-	    }
-	  glEnd ();
-	}
-    }
+    glEnable (GL_COLOR_MATERIAL);
+    glDisable (GL_LIGHTING);
 
+    GLfloat textcol[3] = { 1 - backcolor,
+			   1 - backcolor,
+			   1 - backcolor };
+    glColor3fv (textcol);
 
+    glLineWidth (1.0f);
 
-  glPopMatrix();
-  
-  DrawCoordinateCross ();
-  DrawNetgenLogo ();
+    float len = 1;
+    glBegin(GL_LINES);
+    glVertex3f (0.0f, 0.0f, 0.0f);
+    glVertex3f (len, 0.0f, 0.0f);
+    glVertex3f (0.0f, 0.0f, 0.0f);
+    glVertex3f (0.0f, len, 0.0f);
+    glVertex3f (0.0f, 0.0f, 0.0f);
+    glVertex3f (0.0f, 0.0f, len);
+    glEnd ();
 
-  glFinish();  
-}
 
+    glPushAttrib (GL_LIST_BIT);
+    glListBase (fontbase);
 
-void VisualSceneGeometry2d :: BuildScene (int zoomall)
-{
-  Box<2> bbox;
+    char buf[20];
 
-  geometry2d->GetBoundingBox (bbox);
-  
-  Point<2> c = Center (bbox.PMin(), bbox.PMax());
+    glRasterPos3d (len, 0.0f, 0.0f);
+    sprintf (buf, "x");
+    glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
+    glRasterPos3d (0.0f, len, 0.0f);
+    sprintf (buf, "y");
+    glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
+    glRasterPos3d (0.0f, 0.0f, len);
+    sprintf (buf, "z");
+    glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
 
-  center = Point3d (c(0), c(1), 0);
-  rad = Dist (bbox.PMin(), bbox.PMax()) / 2;
-
-  CalcTransformationMatrices();
-}
+    glPopAttrib ();
+    glEnable (GL_LIGHTING);
 
+    glMatrixMode (GL_PROJECTION); 
+    glPopMatrix();
+    glMatrixMode (GL_MODELVIEW); 
+    glPopMatrix();
+    glEnable (GL_DEPTH_TEST);
+  }
 
 
+  void VisualScene :: DrawNetgenLogo ()
+  {
+    if (!vispar.drawnetgenlogo) return;
 
+    glDisable (GL_DEPTH_TEST);
+    glMatrixMode (GL_PROJECTION); 
+    glPushMatrix();
+    glLoadIdentity();
 
+    glMatrixMode (GL_MODELVIEW); 
+    glPushMatrix();
+    glLoadIdentity();
 
+    GLint viewport[4];
+    glGetIntegerv (GL_VIEWPORT, viewport);
 
 
 
+    glTranslatef (1, -1, 0.0);
+    glScalef (40.0 / viewport[2], 40.0 / viewport[3], 1);
+    glTranslatef (-6.0, 2.0, 0.0);
 
-/* *********************** Draw STL Geometry **************** */
+    glDisable (GL_CLIP_PLANE0);
+    // glDisable (GL_LIGHTING);
 
+    glEnable (GL_COLOR_MATERIAL);
+    GLfloat textcol[3] = { 1 - backcolor,
+			   1 - backcolor,
+			   1 - backcolor };
+    glColor3fv (textcol);
 
-VisualSceneSTLGeometry :: VisualSceneSTLGeometry ()
-  : VisualScene()
-{
-  ;
-}
+    glLineWidth (1.0f);
 
-VisualSceneSTLGeometry :: ~VisualSceneSTLGeometry ()
-{
-  ;
-}
+    glPushAttrib (GL_LIST_BIT);
+    glListBase (fontbase);
 
-void VisualSceneSTLGeometry :: DrawScene ()
-{
-  int i, j, k;
+    char buf[20];
 
-  if (changeval != stlgeometry->GetNT())
-    BuildScene();
+    glRasterPos3d (0.0f, 0.0f, 0.0f);
+    sprintf (buf, "Netgen 4.4");
+    glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
 
-  changeval = stlgeometry->GetNT();
+    glPopAttrib ();
 
+    glMatrixMode (GL_PROJECTION); 
+    glPopMatrix();
+    glMatrixMode (GL_MODELVIEW); 
+    glPopMatrix();
+    glEnable (GL_DEPTH_TEST);
+  }
 
-  glClearColor(backcolor, backcolor, backcolor, 1.0);
-  glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
 
-  SetLight();
 
 
-  glPushMatrix();
-  glMultMatrixf (transformationmat);
 
+  /* *********************** Draw 2D Geometry **************** */
 
 
+  VisualSceneGeometry2d :: VisualSceneGeometry2d ()
+    : VisualScene()
+  {
+    ;
+  }
 
-  glShadeModel (GL_SMOOTH);
-  glDisable (GL_COLOR_MATERIAL);
-  glPolygonMode (GL_FRONT_AND_BACK, GL_FILL);
+  VisualSceneGeometry2d :: ~VisualSceneGeometry2d ()
+  {
+    ;
+  }
 
-  glEnable (GL_BLEND);
-  glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
 
 
-  double shine = vispar.shininess;
-  double transp = vispar.transp;
+  void VisualSceneGeometry2d :: DrawScene ()
+  {
+    if (changeval != geometry2d->GetSplines().Size())
+      BuildScene();
+    changeval = geometry2d->GetSplines().Size();
 
-  glMaterialf (GL_FRONT_AND_BACK, GL_SHININESS, shine);
-  glLogicOp (GL_COPY);
+  
+    glClearColor(backcolor, backcolor, backcolor, 1.0);
+    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
 
+    SetLight();
 
-  float mat_col[] = { 0.2, 0.2, 0.8, 1 };
-  glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col);
+    //  glEnable (GL_LIGHT0);
+    glDisable (GL_LIGHTING);
+    glPushMatrix();
+    glMultMatrixf (transformationmat);
 
-  glPolygonOffset (1, 1);
-  glEnable (GL_POLYGON_OFFSET_FILL);
+    //  SetClippingPlane ();
 
-  glCallList (trilists.Get(1));
+    glShadeModel (GL_SMOOTH);
+    glEnable (GL_COLOR_MATERIAL);
+    glPolygonMode (GL_FRONT_AND_BACK, GL_LINE);
+  
+    //  float mat_col[] = { 0, 0, 1, 1 };
+    //  glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col);
+    glColor3f (0, 0, 1);
+  
 
-  glDisable (GL_POLYGON_OFFSET_FILL);
+    ARRAY<Point<2> > points, otherpoints;
 
+    for (int i = 1; i <= geometry2d->GetSplines().Size(); i++)
+      {
+	geometry2d->GetSplines().Get(i)->GetPoints (20, points);
+      
+	glBegin (GL_LINE_STRIP);
+	for (int j = 0; j < points.Size(); j++)
+	  glVertex3f (points[j](0), points[j](1), 0);
+	glEnd(); 
+      }
 
-  int showtrias = vispar.showstltrias;
+    glColor3f (1, 0, 0);
 
-  if (showtrias)
-    {
-      float mat_coll[] = { 0.2, 0.2, 0.2, 1 };
-      glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_coll);
-      glPolygonMode (GL_FRONT_AND_BACK, GL_LINE);
-      
-      glCallList (trilists.Get(1));
-    }
+    for (int i = 1; i <= geometry2d->GetSplines().Size(); i++)
+      {
+	int other = geometry2d->GetSplines().Get(i)->copyfrom;
+	if (other != -1)
+	  {
+	    geometry2d->GetSplines().Get(i)->GetPoints (6, points);
+	    geometry2d->GetSplines().Get(other)->GetPoints (6, otherpoints);
+	    glBegin (GL_LINES);
+	    for (int j = 1; j < 5; j++)
+	      {
+		glVertex3f (points[j](0), points[j](1), 0);
+		glVertex3f (otherpoints[j](0), otherpoints[j](1), 0);
+	      }
+	    glEnd ();
+	  }
+      }
 
-  /*
 
-  glBegin (GL_TRIANGLES);
-  for (j = 1; j <= stlgeometry -> GetNT(); j++)
-    {
-      const STLTriangle & tria = stlgeometry -> GetTriangle(j);
-      glNormal3f (tria.normal.X(),
-		  tria.normal.Y(),
-		  tria.normal.Z());
-		  
-      for (k = 0; k < 3; k++)
-	{
-	  glVertex3f (tria.pts[k].X(),
-		      tria.pts[k].Y(),
-		      tria.pts[k].Z());
-	}
-    }    
-  glEnd ();
-  */  
 
+    glPopMatrix();
+  
+    DrawCoordinateCross ();
+    DrawNetgenLogo ();
 
+    glFinish();  
+  }
 
- 
-  glPopMatrix();
-  glFinish();  
-}
 
+  void VisualSceneGeometry2d :: BuildScene (int zoomall)
+  {
+    Box<2> bbox;
 
-void VisualSceneSTLGeometry :: BuildScene (int zoomall)
-{
-  int i, j, k;
+    geometry2d->GetBoundingBox (bbox);
   
-  // cout << "rebuild stl geometry scene" << endl;
+    Point<2> c = Center (bbox.PMin(), bbox.PMax());
 
-  center = stlgeometry -> GetBoundingBox().Center();
-  rad = stlgeometry -> GetBoundingBox().Diam() / 2;
+    center = Point3d (c(0), c(1), 0);
+    rad = Dist (bbox.PMin(), bbox.PMax()) / 2;
 
+    CalcTransformationMatrices();
+  }
 
-  CalcTransformationMatrices();
 
-  for (i = 1; i <= trilists.Size(); i++)
-    glDeleteLists (trilists.Elem(i), 1);
-  trilists.SetSize(0);
 
 
-  trilists.Append (glGenLists (1));
-  glNewList (trilists.Last(), GL_COMPILE);
 
-  glEnable (GL_NORMALIZE);
 
 
-  glBegin (GL_TRIANGLES);
-  for (j = 1; j <= stlgeometry -> GetNT(); j++)
-    {
-      const Vec3d & n = stlgeometry->GetTriangle(j).Normal();
-      glNormal3f (n.X(), n.Y(), n.Z());
-      
-      for (k = 1; k <= 3; k++)
-	{
-	  const Point3d & p = 
-	    stlgeometry->GetPoint (stlgeometry -> GetTriangle(j).PNum(k));
-	  glVertex3f (p.X(),p.Y(), p.Z());
- 
-	  //	  int pi = ta.GetTriangle(j).PNum(k);
-	  //	  glNormal3f (ta.GetNormal (pi).X(),
-	  //	      ta.GetNormal (pi).Y(),
-	  //	      ta.GetNormal (pi).Z());
-	  //	  glVertex3f (ta.GetPoint(pi).X(),
-	  //	      ta.GetPoint(pi).Y(),
-	  //	      ta.GetPoint(pi).Z());
-	}
-    }    
-  glEnd ();
-      
-  glEndList ();
-}
 
 
 
-#ifdef OCCGEOMETRY
-/* *********************** Draw OCC Geometry **************** */
+  /* *********************** Draw STL Geometry **************** */
 
 
-VisualSceneOCCGeometry :: VisualSceneOCCGeometry ()
-  : VisualScene()
-{
-  trilists.SetSize(0);
-  linelists.SetSize(1);
-}
+  VisualSceneSTLGeometry :: VisualSceneSTLGeometry ()
+    : VisualScene()
+  {
+    ;
+  }
 
-VisualSceneOCCGeometry :: ~VisualSceneOCCGeometry ()
-{
-  ;
-}
+  VisualSceneSTLGeometry :: ~VisualSceneSTLGeometry ()
+  {
+    ;
+  }
 
-void VisualSceneOCCGeometry :: DrawScene ()
-{
-  int i, j, k;
+  void VisualSceneSTLGeometry :: DrawScene ()
+  {
+    if (changeval != stlgeometry->GetNT())
+      BuildScene();
 
+    changeval = stlgeometry->GetNT();
 
-  if ( occgeometry->changed )
-    {
-      BuildScene();
-      occgeometry -> changed = 0;
-    }
 
-  glClearColor(backcolor, backcolor, backcolor, 1.0);
-  glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
-  SetLight();
+    glClearColor(backcolor, backcolor, backcolor, 1.0);
+    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
 
-  //  glEnable (GL_COLOR_MATERIAL);
+    SetLight();
 
-  //  glDisable (GL_SHADING);
-  //  glColor3f (0.0f, 1.0f, 1.0f);
-  //  glLineWidth (1.0f);
-  //  glShadeModel (GL_SMOOTH);
 
-  //  glCallList (linelists.Get(1));
+    glPushMatrix();
+    glMultMatrixf (transformationmat);
 
-  //  SetLight();
 
-  glPushMatrix();
-  glMultMatrixf (transformationmat);
 
-  glShadeModel (GL_SMOOTH);
-  glDisable (GL_COLOR_MATERIAL);
-  glPolygonMode (GL_FRONT_AND_BACK, GL_FILL);
 
-  glEnable (GL_BLEND);
-  glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
+    glShadeModel (GL_SMOOTH);
+    glDisable (GL_COLOR_MATERIAL);
+    glPolygonMode (GL_FRONT_AND_BACK, GL_FILL);
 
-  //  glEnable (GL_LIGHTING);
+    glEnable (GL_BLEND);
+    glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
 
-  double shine = vispar.shininess;
-  double transp = vispar.transp;
 
-  glMaterialf (GL_FRONT_AND_BACK, GL_SHININESS, shine);
-  glLogicOp (GL_COPY);
+    double shine = vispar.shininess;
+    // double transp = vispar.transp;
 
+    glMaterialf (GL_FRONT_AND_BACK, GL_SHININESS, shine);
+    glLogicOp (GL_COPY);
 
-  float mat_col[] = { 0.2, 0.2, 0.8, 1 };
-  glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col);
 
-  glPolygonOffset (1, 1);
-  glEnable (GL_POLYGON_OFFSET_FILL);
+    float mat_col[] = { 0.2f, 0.2f, 0.8f, 1.0f};
+    glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col);
 
-  glCallList (linelists.Get(1));
+    glPolygonOffset (1, 1);
+    glEnable (GL_POLYGON_OFFSET_FILL);
 
-  glDisable (GL_POLYGON_OFFSET_FILL);
- 
-  glPopMatrix();
-  DrawCoordinateCross ();
-  DrawNetgenLogo ();
-  glFinish();  
+    glCallList (trilists.Get(1));
 
-  glDisable (GL_POLYGON_OFFSET_FILL);
+    glDisable (GL_POLYGON_OFFSET_FILL);
 
-}
 
+    int showtrias = vispar.showstltrias;
 
-void VisualSceneOCCGeometry :: BuildScene (int zoomall)
-{
-  int i = 0, j, k;
-  
-  Box3d box;
+    if (showtrias)
+      {
+	float mat_coll[] = { 0.2f, 0.2f, 0.2f, 1.0f };
+	glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_coll);
+	glPolygonMode (GL_FRONT_AND_BACK, GL_LINE);
+      
+	glCallList (trilists.Get(1));
+      }
 
-  TopExp_Explorer ex, ex_edge;
+    /*
 
-  for (ex.Init(occgeometry -> shape, TopAbs_VERTEX); ex.More(); ex.Next())
+    glBegin (GL_TRIANGLES);
+    for (j = 1; j <= stlgeometry -> GetNT(); j++)
     {
-      i++;
-      gp_Pnt p = BRep_Tool::Pnt (TopoDS::Vertex(ex.Current()));
-
-      if (i == 1) 
-	box.SetPoint (Point<3>(p.X(), p.Y(), p.Z()));
-      else
-	box.AddPoint (Point<3>(p.X(), p.Y(), p.Z()));
+    const STLTriangle & tria = stlgeometry -> GetTriangle(j);
+    glNormal3f (tria.normal.X(),
+    tria.normal.Y(),
+    tria.normal.Z());
+		  
+    for (k = 0; k < 3; k++)
+    {
+    glVertex3f (tria.pts[k].X(),
+    tria.pts[k].Y(),
+    tria.pts[k].Z());
     }
+    }    
+    glEnd ();
+    */  
 
-  center = box.CalcCenter();
-  rad = box.CalcDiam() / 2;
-  
-  CalcTransformationMatrices();
-
-
-  linelists.Elem(1) = glGenLists (1);
-  glNewList (linelists.Last(), GL_COMPILE);
 
-  for (ex_edge.Init(occgeometry -> shape, TopAbs_EDGE);
-       ex_edge.More(); ex_edge.Next())
-    {
-      if (BRep_Tool::Degenerated(TopoDS::Edge(ex_edge.Current()))) continue;
 
-      glBegin (GL_LINE_STRIP);
+ 
+    glPopMatrix();
+    glFinish();  
+  }
 
-      double s0, s1, s;
-      Handle(Geom_Curve) c = BRep_Tool::Curve(TopoDS::Edge(ex_edge.Current()), s0, s1);
-      double d = s1-s0;
-      int i;
-      
-      for (i = 0; i<=50; i++)
-	{
-	  s = s0 + i*d/50;
-	  gp_Pnt p = c->Value (s);
-	  glVertex3f (p.X(),p.Y(), p.Z());
-	}
 
-      glEnd ();
-    }
+  void VisualSceneSTLGeometry :: BuildScene (int zoomall)
+  {
+    //  cout << "rebuild stl geometry scene" << endl;
 
-  //    glPolygonMode (GL_FRONT_AND_BACK, GL_LINE);
+    center = stlgeometry -> GetBoundingBox().Center();
+    rad = stlgeometry -> GetBoundingBox().Diam() / 2;
 
-  glBegin (GL_TRIANGLES);
-  //  int nfaces = occgeometry -> fmap.Extent();
 
-  i = 0;
+    CalcTransformationMatrices();
 
-  TopExp_Explorer exp0, exp1, exp2, exp3;
+    for (int i = 1; i <= trilists.Size(); i++)
+      glDeleteLists (trilists.Elem(i), 1);
+    trilists.SetSize(0);
 
-  //  BRepMesh::Mesh (occgeometry -> shape, occgeometry -> MaxSize()/100);
 
-  for (exp0.Init(occgeometry -> shape, TopAbs_SOLID); exp0.More(); exp0.Next())
-    for (exp1.Init(exp0.Current(), TopAbs_SHELL); exp1.More(); exp1.Next())
-      for (exp2.Init(exp1.Current().Composed(exp0.Current().Orientation()), TopAbs_FACE); exp2.More(); exp2.Next())
-	{
-	  TopoDS_Face face = TopoDS::Face (exp2.Current().Composed(exp1.Current().Orientation()));
+    trilists.Append (glGenLists (1));
+    glNewList (trilists.Last(), GL_COMPILE);
 
-	  TopLoc_Location loc;
-	  Handle(Geom_Surface) surf = BRep_Tool::Surface (face);
-	  BRepAdaptor_Surface sf(face, Standard_False);
-	  BRepLProp_SLProps prop(sf, 1, 1e-5);
+    glEnable (GL_NORMALIZE);
 
-	  //	  BRepMesh::Mesh (face, 1e-3);
-	  (*testout) << "meshing face " << ++i << endl;
+    glBegin (GL_TRIANGLES);
+    for (int j = 1; j <= stlgeometry -> GetNT(); j++)
+      {
+	const Vec3d & n = stlgeometry->GetTriangle(j).Normal();
+	glNormal3f (n.X(), n.Y(), n.Z());
+      
+	for (int k = 1; k <= 3; k++)
 	  {
-	    int edges = 0;
-	    for (exp3.Init (face, TopAbs_EDGE); exp3.More(); exp3.Next())
-	      {
-		edges++;
-		(*testout) << "edge " << edges;
-		if (BRep_Tool::Degenerated(TopoDS::Edge(exp3.Current())))
-		  (*testout) << " degenerated";
-		(*testout) << endl;
-	      }
+	    const Point3d & p = 
+	      stlgeometry->GetPoint (stlgeometry -> GetTriangle(j).PNum(k));
+	    glVertex3f (p.X(),p.Y(), p.Z());
 	  }
-	  BRepMesh::Mesh (face, occgeometry -> MaxSize()/100);
-	  (*testout) << "done" << endl;
-	  //	  cout << "Warning: mvdraw fixed meshing size for visualization" << endl;
-	  // cout << "face " << ++i << endl;
-
-	  Handle(Poly_Triangulation) triangulation = BRep_Tool::Triangulation (face, loc);
-	  if (!triangulation) continue;
-	  
-	  int ntriangles = triangulation -> NbTriangles();
-	  //	  cout << "nbtriangles = " << ntriangles << endl;
-	  for (j = 1; j <= ntriangles; j++)
-	    {
-	      // (*testout) << "drawing trig " << j << "/" << ntriangles << " " << flush;
-	      //	      cout << "j = " << j << endl;
-	      Poly_Triangle triangle = (triangulation -> Triangles())(j);
-	      for (k = 1; k <= 3; k++)
-		{
-		  (*testout) << "." << flush;
-		  gp_Pnt2d uv = (triangulation -> UVNodes())(triangle(k));
-		  gp_Pnt pnt;
-		  gp_Vec du, dv;
-		  prop.SetParameters (uv.X(), uv.Y());
-		  surf->D0 (uv.X(), uv.Y(), pnt);
-		  gp_Vec n;
-
-		  if (prop.IsNormalDefined())
-		    {
-		      n = prop.Normal();
-		    }
-		  else
-		    n = gp_Vec (0,0,0);
-
-
-		  //		  surf->D1 (uv.X(), uv.Y(), pnt, du, dv);
-		  //		  gp_Vec n = du.Crossed(dv).Normalized();
-		  if (face.Orientation() == TopAbs_REVERSED) n *= -1;
-		  glNormal3f (n.X(), n.Y(), n.Z());
-		  glVertex3f (pnt.X(), pnt.Y(), pnt.Z());
-		}
-	      (*testout) << endl;
-	    }
-	}
-  
-  glEnd ();
+      }    
+    glEnd ();
+      
+    glEndList ();
+  }
 
 
 
-  glEndList ();
 
-}
 
-#endif
 
 
 
-#ifdef STEP
 
-/* *********************** Draw STEP Geometry **************** */
 
 
-VisualSceneSTEPGeometry :: VisualSceneSTEPGeometry ()
-  : VisualScene ()
-{
-  gllists.SetSize(0);
-}
 
+  VisualSceneSpecPoints :: VisualSceneSpecPoints ()
+    : VisualScene()
+  {
+    ;
+  }
 
-VisualSceneSTEPGeometry :: ~VisualSceneSTEPGeometry ()
-{
-  cout << "VisualSceneSTEPGeometry::~VisualSceneSTEPGeometry called "
-       << "but not implemented" << endl;
-}
-  
+  VisualSceneSpecPoints :: ~VisualSceneSpecPoints ()
+  {
+    ;
+  }
 
-void VisualSceneSTEPGeometry :: BuildScene (int zoomall)
-{
-  cout << "VisualSceneSTEPGeometry::BuildScene called" << endl;
 
-  const VisualApproximation & visapprox = 
-    stepgeometry->GetVisualApproximation();
+  void VisualSceneSpecPoints :: DrawScene ()
+  {
+    if (!mesh) 
+      {
+	VisualScene::DrawScene();
+	return;
+      }
 
-  INDEX i, k, pi;
+    if (changeval != specpoints.Size())
+      BuildScene();
+    changeval = specpoints.Size();
 
-  Box3d box;
-  int hasp = 0;
 
-  for (i = 1; i <= visapprox.GetNP(); i++)      
-    {
-      if (hasp)
-	box.AddPoint (visapprox.GetPoint(i));
-      else
-	{
-	  hasp = 1;
-	  box.SetPoint (visapprox.GetPoint(i));
-	}
-    }
-  if (hasp)
-    {
-      center = box.CalcCenter();
-      rad = box.CalcDiam() / 2;
-    }
-
-  cout << "center " << center << endl
-       << "radius " << rad << endl;
 
-  CalcTransformationMatrices();
+    glClearColor(backcolor, backcolor, backcolor, 1.0);
+    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
 
-  for (i = 1; i <= gllists.Size(); i++)
-    glDeleteLists (gllists.Elem(i), 1);
-  gllists.SetSize(0);
+    glEnable (GL_COLOR_MATERIAL);
+    glColor3f (1.0f, 1.0f, 1.0f);
+    glLineWidth (1.0f);
 
-  // build list of points
-  gllists.Append (glGenLists (1));
-  glNewList (gllists.Last(), GL_COMPILE);
-   
-  glPointSize( 3.0 );
+    glPushMatrix();
+    glMultMatrixf (transformationmat);
 
-  glBegin( GL_POINTS );
-    for (i = 1; i <= visapprox.GetNP(); i++)
+    //  glEnable (GL_COLOR);
+    //  glDisable (GL_COLOR_MATERIAL);
+    if (vispar.drawedtangents)
       {
-	const Point3d & p = visapprox.GetPoint(i);
-        glVertex3f( p.X(), p.Y(), p.Z());
+	glColor3d (1, 0, 0);
+	glBegin (GL_LINES);
+	for (int i = 1; i <= specpoints.Size(); i++)
+	  {
+	    const Point3d p1 = specpoints.Get(i).p;
+	    const Point3d p2 = specpoints.Get(i).p + len * specpoints.Get(i).v;
+	    glVertex3d (p1.X(), p1.Y(), p1.Z());
+	    glVertex3d (p2.X(), p2.Y(), p2.Z());
+	  }
+	glEnd();
       }
-  glEnd();
-
-  glEndList();
 
-  // build list of segments
-  gllists.Append (glGenLists (1));
-  glNewList (gllists.Last(), GL_COMPILE);
-   
-  glLineWidth( 1.0f );
-
-  glBegin(GL_LINES);
-    for (i = 1; i <= visapprox.GetNSeg(); i++)
+    if (vispar.drawededges)
       {
-	for (k = 1; k <= 2; k++)
+	glColor3d (1, 0, 0);
+	glBegin (GL_LINES);
+	for (int i = 1; i <= mesh->GetNSeg(); i++)
 	  {
-	    pi = visapprox.GetSegment(i).PNum(k);
-	    glVertex3f(visapprox.GetPoint(pi).X(),
-		       visapprox.GetPoint(pi).Y(),
-		       visapprox.GetPoint(pi).Z());
+	    const Segment & seg = mesh -> LineSegment (i);
+	    glVertex3dv ( &(*mesh)[seg.p1].X() );
+	    glVertex3dv ( &(*mesh)[seg.p2].X() );
 	  }
-      }    
-  glEnd();
-
-  glEndList();
-}
-
-
-void VisualSceneSTEPGeometry :: DrawScene ()
-{
-  if (! stepgeometry)
-    {
-      cerr << "VisualSceneSTEPGeometry::DrawScene: no STEP geometry" << endl;
-      return;
-    }
-
-  const VisualApproximation & visapprox = 
-    stepgeometry -> GetVisualApproximation();
-  if (changeval != visapprox.GetNP())
-    BuildScene();
-
-  changeval = visapprox.GetNP();
-
-  glClearColor(backcolor, backcolor, backcolor, 1.0);
-  glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
-
-  glPushMatrix();
-  glMultMatrixf (transformationmat);
-
-  // draw segments
-//   float mat_coll[] = { 0.2, 0.2, 0.2, 1 };
-//   glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_coll);
-//   glPolygonMode (GL_FRONT_AND_BACK, GL_LINE);
-
-  glEnable (GL_COLOR_MATERIAL);
-  glColor3f(0.0, 1.0, 0.0);
-  glCallList (gllists.Get(2));
-  glColor3f (1.0, 0.0, 0.0);
-  glCallList (gllists.Get(1));
-
-  glPopMatrix();
-  glFinish();
-}
-
-#endif
-
-
-
-
-
-
-
-
-
-
-VisualSceneSpecPoints :: VisualSceneSpecPoints ()
-  : VisualScene()
-{
-  ;
-}
-
-VisualSceneSpecPoints :: ~VisualSceneSpecPoints ()
-{
-  ;
-}
-
-
-void VisualSceneSpecPoints :: DrawScene ()
-{
-  int i, j, k;
-
-
-  if (!mesh) 
-    {
-      VisualScene::DrawScene();
-      return;
-    }
-
-  if (changeval != specpoints.Size())
-    BuildScene();
-  changeval = specpoints.Size();
-
-
-
-  glClearColor(backcolor, backcolor, backcolor, 1.0);
-  glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
-
-  glEnable (GL_COLOR_MATERIAL);
-  glColor3f (1.0f, 1.0f, 1.0f);
-  glLineWidth (1.0f);
-
-  glPushMatrix();
-  glMultMatrixf (transformationmat);
-
-  //  glEnable (GL_COLOR);
-  //  glDisable (GL_COLOR_MATERIAL);
-  if (vispar.drawedtangents)
-    {
-      glColor3d (1, 0, 0);
-      glBegin (GL_LINES);
-      for (i = 1; i <= specpoints.Size(); i++)
-	{
-	  const Point3d p1 = specpoints.Get(i).p;
-	  const Point3d p2 = specpoints.Get(i).p + len * specpoints.Get(i).v;
-	  glVertex3d (p1.X(), p1.Y(), p1.Z());
-	  glVertex3d (p2.X(), p2.Y(), p2.Z());
-	}
-      glEnd();
-    }
-
-  if (vispar.drawededges)
-    {
-      glColor3d (1, 0, 0);
-      glBegin (GL_LINES);
-      for (i = 1; i <= mesh->GetNSeg(); i++)
-	{
-	  const Segment & seg = mesh -> LineSegment (i);
-	  const Point3d p1 = mesh -> Point (seg.p1);
-	  const Point3d p2 = mesh -> Point (seg.p2);
-	  glVertex3d (p1.X(), p1.Y(), p1.Z());
-	  glVertex3d (p2.X(), p2.Y(), p2.Z());
-	}
-      glEnd();
-    }
+	glEnd();
+      }
 
-  if (vispar.drawededgenrs)
-    {
-      glEnable (GL_COLOR_MATERIAL);
-      GLfloat textcol[3] = { 1 - backcolor,
+    if (vispar.drawededgenrs)
+      {
+	glEnable (GL_COLOR_MATERIAL);
+	GLfloat textcol[3] = { 1 - backcolor,
 			       1 - backcolor,
 			       1 - backcolor };
-      glColor3fv (textcol);
-      glNormal3d (0, 0, 1);
-      glPushAttrib (GL_LIST_BIT);
-      glListBase (fontbase);
+	glColor3fv (textcol);
+	glNormal3d (0, 0, 1);
+	glPushAttrib (GL_LIST_BIT);
+	glListBase (fontbase);
 
-      char buf[20];
-      for (i = 1; i <= mesh->GetNSeg(); i++)
-	{
-	  const Segment & seg = mesh -> LineSegment (i);
-	  const Point3d p1 = mesh -> Point (seg.p1);
-	  const Point3d p2 = mesh -> Point (seg.p2);
+	char buf[20];
+	for (int i = 1; i <= mesh->GetNSeg(); i++)
+	  {
+	    const Segment & seg = mesh -> LineSegment (i);
+	    const Point3d p1 = mesh -> Point (seg.p1);
+	    const Point3d p2 = mesh -> Point (seg.p2);
 
-	  const Point3d p = Center (p1, p2);
-	  glRasterPos3d (p.X(), p.Y(), p.Z());
+	    const Point3d p = Center (p1, p2);
+	    glRasterPos3d (p.X(), p.Y(), p.Z());
 	  
-	  sprintf (buf, "%d", seg.edgenr);
-	  glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
-	}
+	    sprintf (buf, "%d", seg.edgenr);
+	    glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
+	  }
       
-      glPopAttrib ();
-      glDisable (GL_COLOR_MATERIAL);
-    }
+	glPopAttrib ();
+	glDisable (GL_COLOR_MATERIAL);
+      }
 
 
-  if (vispar.drawedpoints)
-    {
-      glColor3d (0, 0, 1);
-      glPointSize( 3.0 );
+    if (vispar.drawedpoints)
+      {
+	glColor3d (0, 0, 1);
+	glPointSize( 3.0 );
 
-      glBegin( GL_POINTS );
-      for (i = 1; i <= mesh -> GetNP(); i++)
-	{
-	  const Point3d & p = mesh -> Point(i);
-	  glVertex3f( p.X(), p.Y(), p.Z());
-	}
-      glEnd();
-    }
+	/*
+	  float range[2];
+	  glGetFloatv(GL_POINT_SIZE_RANGE, &range[0]);
+	  cout << "max ptsize = " << range[0] << "-" << range[1] << endl;
+	*/
+      
 
-  if (vispar.drawedpointnrs)
-    {
-      glEnable (GL_COLOR_MATERIAL);
-      GLfloat textcol[3] = { 1 - backcolor,
+	glBegin( GL_POINTS );
+	for (int i = 1; i <= mesh -> GetNP(); i++)
+	  {
+	    const Point3d & p = mesh -> Point(i);
+	    if (i % 2)
+	      glVertex3f( p.X(), p.Y(), p.Z());
+	  }
+	glEnd();
+
+	static GLubyte knoedel[] = 
+	  {
+	    0xfe, 0xfe, 0xfe, 0xfe, 0xfe, 0xfe, 0xfe,
+	  };
+	glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
+
+	glDisable (GL_COLOR_MATERIAL);
+	glDisable (GL_LIGHTING);
+	glDisable (GL_CLIP_PLANE0);
+      
+	for (int i = 1; i <= mesh -> GetNP(); i++)
+	  {
+	    const Point3d & p = mesh -> Point(i);
+	    glRasterPos3d (p.X(), p.Y(), p.Z());
+	    glBitmap (7, 7, 3, 3, 0, 0, &knoedel[0]);
+	  }
+      
+      }
+
+    if (vispar.drawedpointnrs)
+      {
+	glEnable (GL_COLOR_MATERIAL);
+	GLfloat textcol[3] = { 1 - backcolor,
 			       1 - backcolor,
 			       1 - backcolor };
-      glColor3fv (textcol);
-      glNormal3d (0, 0, 1);
-      glPushAttrib (GL_LIST_BIT);
-      glListBase (fontbase);
+	glColor3fv (textcol);
+	glNormal3d (0, 0, 1);
+	glPushAttrib (GL_LIST_BIT);
+	glListBase (fontbase);
       
-      char buf[20];
-      for (i = 1; i <= mesh->GetNP(); i++)
-	{
-	  const Point3d & p = mesh->Point(i);
-	  glRasterPos3d (p.X(), p.Y(), p.Z());
+	char buf[20];
+	for (int i = 1; i <= mesh->GetNP(); i++)
+	  {
+	    const Point3d & p = mesh->Point(i);
+	    glRasterPos3d (p.X(), p.Y(), p.Z());
 	  
-	  sprintf (buf, "%d", i);
-	  glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
-	}
+	    sprintf (buf, "%d", i);
+	    glCallLists (strlen (buf), GL_UNSIGNED_BYTE, buf);
+	  }
       
-      glPopAttrib ();
-      glDisable (GL_COLOR_MATERIAL);
-    }
+	glPopAttrib ();
+	glDisable (GL_COLOR_MATERIAL);
+      }
 
 
-  glPopMatrix();
+    glPopMatrix();
 
-  if (vispar.drawcoordinatecross)
-    DrawCoordinateCross ();
-  DrawNetgenLogo ();
+    if (vispar.drawcoordinatecross)
+      DrawCoordinateCross ();
+    DrawNetgenLogo ();
 
-  glFinish();  
-}
+    glFinish();  
+  }
 
 
-void VisualSceneSpecPoints :: BuildScene (int zoomall)
-{
-  int i, j, k;
+  void VisualSceneSpecPoints :: BuildScene (int zoomall)
+  {
+    if (!mesh) 
+      {
+	VisualScene::BuildScene(zoomall);
+	return;
+      }
   
-  if (!mesh) 
-    {
-      VisualScene::BuildScene(zoomall);
-      return;
-    }
-
-  Box3d box;
-
-  if (mesh->GetNSeg())
-    {
-      box.SetPoint (mesh->Point (mesh->LineSegment(1).p1));
-      for (i = 1; i <= mesh->GetNSeg(); i++)
-	{
-	  box.AddPoint (mesh->Point (mesh->LineSegment(i).p1));
-	  box.AddPoint (mesh->Point (mesh->LineSegment(i).p2));
-	}
-    }
-  else if (specpoints.Size() >= 2)
-    {
-      box.SetPoint (specpoints.Get(1).p);
-      for (i = 2; i <= specpoints.Size(); i++)
-	box.AddPoint (specpoints.Get(i).p);
-    }
-  else
-    {
-      box = Box3d (Point3d (0,0,0), Point3d (1,1,1));
-    }
+    Box3d box;
   
-  if (zoomall == 2 && vispar.centerpoint >= 1 && vispar.centerpoint <= mesh->GetNP())
-    center = mesh->Point (vispar.centerpoint);
-  else
-    center = Center (box.PMin(), box.PMax());
-
-  rad = 0.5 * Dist (box.PMin(), box.PMax());
+    if (mesh->GetNSeg())
+      {
+	box.SetPoint (mesh->Point (mesh->LineSegment(1).p1));
+	for (int i = 1; i <= mesh->GetNSeg(); i++)
+	  {
+	    box.AddPoint (mesh->Point (mesh->LineSegment(i).p1));
+	    box.AddPoint (mesh->Point (mesh->LineSegment(i).p2));
+	  }
+      }
+    else if (specpoints.Size() >= 2)
+      {
+	box.SetPoint (specpoints.Get(1).p);
+	for (int i = 2; i <= specpoints.Size(); i++)
+	  box.AddPoint (specpoints.Get(i).p);
+      }
+    else
+      {
+	box = Box3d (Point3d (0,0,0), Point3d (1,1,1));
+      }
   
+    if (zoomall == 2 && ((vispar.centerpoint >= 1 && vispar.centerpoint <= mesh->GetNP()) ||
+			 vispar.use_center_coords))
+      {
+	if (vispar.use_center_coords)
+	  {
+	    center.X() = vispar.centerx; center.Y() = vispar.centery; center.Z() = vispar.centerz; 
+	  }
+	else
+	  center = mesh->Point (vispar.centerpoint);
+      }
+    else
+      center = Center (box.PMin(), box.PMax());
+        
 
-  CalcTransformationMatrices();
-}
+    rad = 0.5 * Dist (box.PMin(), box.PMax());
+  
+  
+    CalcTransformationMatrices();
+  }
 
 }
diff --git a/Netgen/libsrc/visualization/mvdraw.hpp b/Netgen/libsrc/visualization/mvdraw.hpp
index 691ac246cb..5665533979 100644
--- a/Netgen/libsrc/visualization/mvdraw.hpp
+++ b/Netgen/libsrc/visualization/mvdraw.hpp
@@ -1,6 +1,10 @@
 #ifndef FILE_MVDRAW
 #define FILE_MVDRAW
 
+#include "vispar.hpp"
+
+/*
+
 class VisualizationParameters
 {
 public:
@@ -61,6 +65,13 @@ public:
   int stlchartnumber;
   int stlchartnumberoffset;
 
+  // occ:
+  int occshowvolumenr;
+  bool occshowsurfaces;
+  bool occshowedges;
+  bool occvisproblemfaces;
+  bool occzoomtohighlightedentity;
+
   bool whitebackground;
   int stereo;
   bool usedispllists;
@@ -73,6 +84,10 @@ public:
   VisualizationParameters();
 };
 extern VisualizationParameters vispar;
+*/
+
+
+
 
 
 
@@ -100,7 +115,7 @@ protected:
   GLdouble clipplane[4];
 
   int changeval;
-  static double backcolor;
+  static GLdouble backcolor;
 
 public:
   static GLuint fontbase;
@@ -120,6 +135,8 @@ public:
   
   void CalcTransformationMatrices();
   void StandardRotation (const char * dir);
+  void ArbitraryRotation (const ARRAY<double> & alpha, const ARRAY<Vec3d> & vec);
+  void ArbitraryRotation (const double alpha, const Vec3d & vec);
 
   void MouseMove(int oldx, int oldy,
 		 int newx, int newy,
@@ -136,8 +153,8 @@ public:
   static void SetBackGroundColor (double col)
     { backcolor = col; }
 
-  void CreateTexture (int ncols, int linear);
-  void DrawColorBar (double minval, double maxval, int logscale = 0);
+  void CreateTexture (int ncols, int linear, int typ = GL_DECAL);
+  void DrawColorBar (double minval, double maxval, int logscale = 0, bool linear = 1);
   void DrawCoordinateCross ();
   void DrawNetgenLogo ();
   void SetOpenGlColor(double val, double valmin, double valmax, int logscale = 0);
@@ -195,6 +212,7 @@ public:
 
   virtual void BuildScene (int zoomall = 0);
   virtual void DrawScene ();
+  virtual void MouseDblClick (int px, int py);
 };
 #endif
 
diff --git a/Netgen/libsrc/visualization/soldata.hpp b/Netgen/libsrc/visualization/soldata.hpp
index b2c11cfbd0..44e0d0a56e 100644
--- a/Netgen/libsrc/visualization/soldata.hpp
+++ b/Netgen/libsrc/visualization/soldata.hpp
@@ -2,18 +2,20 @@
 #define FILE_SOLDATA
 
 
+using namespace std;
 
 class SolutionData
 {
 protected:
-  std::string name;
+
+  string name;
   int components;
   bool iscomplex;
 
-  // int dist;
-  // int order;
+  int multidimcomponent;
+
 public:
-  SolutionData (const std::string & aname, 
+  SolutionData (const string & aname, 
 		int acomponents = 1, bool aiscomplex = 0)
     : name(aname), components(acomponents), iscomplex(aiscomplex)
   { ; }
@@ -24,16 +26,18 @@ public:
   int GetComponents() { return components; }
   bool IsComplex() { return iscomplex; }
 
-  virtual bool GetValue (int elnr, 
-			 double lam1, double lam2, double lam3,
-			 double * values) 
-  { return 0; }
+  virtual bool GetValue (int /* elnr */, 
+			 double /* lam1 */, double /* lam2 */, double /* lam3 */,
+			 double * /* values */) 
+  { return false; }
 
-  virtual bool GetSurfValue (int selnr,
-			     double lam1, double lam2, 
-			     double * values)
-  { return 0; }
+  virtual bool GetSurfValue (int /* selnr */,
+			     double /* lam1 */, double /* lam2 */, 
+			     double * /* values */)
+  { return false; }
 
+  void SetMultiDimComponent (int mc)
+  { multidimcomponent = mc; }
 };
 
 
diff --git a/Netgen/libsrc/visualization/stlmeshing.cpp b/Netgen/libsrc/visualization/stlmeshing.cpp
index 3fdbbadaf7..45adac9d75 100644
--- a/Netgen/libsrc/visualization/stlmeshing.cpp
+++ b/Netgen/libsrc/visualization/stlmeshing.cpp
@@ -102,33 +102,33 @@ void VisualSceneSTLMeshing :: DrawScene ()
   glMaterialf (GL_FRONT_AND_BACK, GL_SHININESS, shine);
   glLogicOp (GL_COPY);
 
-  float mat_colred[] = { 0.9, 0.0, 0.0, 1 };
-  float mat_colgreen[] = { 0.0, 0.9, 0.0, 1 };
-  float mat_colblue[] = { 0.1, 0.1, 1., 1 };
+  float mat_colred[]    = { 0.9f, 0.0f, 0.0f, 1.0f };
+  float mat_colgreen[]  = { 0.0f, 0.9f, 0.0f, 1.0f };
+  float mat_colblue[]   = { 0.1f, 0.1f, 1.0f, 1.0f };
 
-  float mat_colbluegreen[] = { 0.1, 0.5, 0.9, 1 };
-  float mat_colpink[] = { 1., 0.1, 0.5, 1 };
-  float mat_colviolet[] = { 1., 0.1, 1., 1 };
-  float mat_colbrown[] = { 0.8, 0.6, 0.1, 1 };
-  float mat_colorange[] = { 0.9, 0.7, 0.1, 1 };
-  float mat_colturquis[] = { 0.0, 1., 0.8, 1 };
+  float mat_colbluegreen[] = { 0.1f, 0.5f, 0.9f, 1.0f };
+  float mat_colpink[]      = { 1.0f, 0.1f, 0.5f, 1.0f };
+  float mat_colviolet[]    = { 1.0f, 0.1f, 1.0f, 1.0f };
+  float mat_colbrown[]     = { 0.8f, 0.6f, 0.1f, 1.0f };
+  float mat_colorange[]    = { 0.9f, 0.7f, 0.1f, 1.0f };
+  float mat_colturquis[]   = { 0.0f, 1.0f, 0.8f, 1.0f };
 
-  float mat_colgrey[] = { 0.3, 0.3, 0.3, 1 };
+  float mat_colgrey[] = { 0.3f, 0.3f, 0.3f, 1.0f };
 
-  float mat_collred[] = { 1., 0.5, 0.5, 1 };
-  float mat_collgreen[] = { 0.2, 1., 0.2, 1 };
-  float mat_collbrown[] = { 1., 0.8, 0.3, 1 };
+  float mat_collred[]   = { 1.0f, 0.5f, 0.5f, 1.0f };
+  float mat_collgreen[] = { 0.2f, 1.9f, 0.2f, 1.0f };
+  float mat_collbrown[] = { 1.0f, 0.8f, 0.3f, 1.0f };
 
-  float mat_collgrey[] = { 0.8, 0.8, 0.8, 1 };
-  float mat_colmgrey[] = { 0.4, 0.4, 0.4, 1 };
+  float mat_collgrey[] = { 0.8f, 0.8f, 0.8f, 1.0f };
+  float mat_colmgrey[] = { 0.4f, 0.4f, 0.4f, 1.0f };
 
-  float mat_colstlbody[] = { 0., 0., 0.8, 1 };
-  float mat_colseltrig[] = { 0.7, 0.7, 0.3, 1 };
-  float mat_colseledge[] = { 0.7, 0.7, 1., 1 };
+  float mat_colstlbody[] = { 0.0f, 0.0f, 0.8f, 1.0f };
+  float mat_colseltrig[] = { 0.7f, 0.7f, 0.3f, 1.0f };
+  float mat_colseledge[] = { 0.7f, 0.7f, 1.0f, 1.0f };
 
   glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_colblue);
 
-  float pgoff = 0.5;
+  float pgoff = 0.5f;
 
   glPolygonOffset (pgoff*1, pgoff*1);
   glEnable (GL_POLYGON_OFFSET_FILL);
@@ -809,14 +809,11 @@ void VisualSceneSTLMeshing :: DrawScene ()
 
 void VisualSceneSTLMeshing :: BuildScene (int zoomall)
 {
-  int i, j, k;
-
   if (selecttrig && zoomall == 2)
-    {
-      center = stlgeometry -> GetPoint ( stlgeometry->GetTriangle(selecttrig).PNum(nodeofseltrig));
-    }
+    center = stlgeometry -> GetPoint ( stlgeometry->GetTriangle(selecttrig).PNum(nodeofseltrig));
   else
     center = stlgeometry -> GetBoundingBox().Center();
+
   rad = stlgeometry -> GetBoundingBox().Diam() / 2;
 
   CalcTransformationMatrices();
@@ -1067,9 +1064,6 @@ void VisualSceneSTLMeshing :: MouseDblClick (int px, int py)
 
 
 
-
-
-
 VisualSceneSTLMeshing vsstlmeshing;
 
 #endif
diff --git a/Netgen/libsrc/visualization/vispar.hpp b/Netgen/libsrc/visualization/vispar.hpp
new file mode 100644
index 0000000000..3a35294182
--- /dev/null
+++ b/Netgen/libsrc/visualization/vispar.hpp
@@ -0,0 +1,89 @@
+#ifndef FILE_VISPAR
+#define FILE_VISPAR
+
+class VisualizationParameters
+{
+public:
+  double lightamb;
+  double lightdiff;
+  double lightspec;
+  double shininess;
+  double transp;
+  int locviewer;
+  char selectvisual[20];
+  int showstltrias;
+  
+  Vec3d clipnormal;
+  double clipdist;
+  int clipenable;
+  int clipplanetimestamp;
+
+  int colormeshsize;
+
+  int drawfilledtrigs;
+  int drawbadels;
+  int drawoutline;
+  int drawedges;
+  int subdivisions;
+
+  int drawprisms;
+  int drawpyramids;
+  int drawhexes;
+  double shrink;
+  int drawidentified;
+  int drawpointnumbers;
+  int drawedgenumbers;
+  int drawfacenumbers;
+  int drawelementnumbers;
+  int drawdomainsurf;
+  int drawtets;
+  int drawtetsdomain;
+
+  int drawededges;
+  int drawedpoints;
+  int drawedpointnrs;
+  int drawedtangents;
+  int drawededgenrs;
+  int drawmetispartition;
+
+  int drawcurveproj;
+  int drawcurveprojedge;
+  
+
+  int centerpoint;
+  int drawelement;
+
+  // stl:
+  int stlshowtrias;
+  int stlshowfilledtrias;
+  int stlshowedges;
+  int stlshowmarktrias;
+  int stlshowactivechart;
+  int stlchartnumber;
+  int stlchartnumberoffset;
+
+  // occ:
+  int occshowvolumenr;
+  bool occshowsurfaces;
+  bool occshowedges;
+  bool occvisproblemfaces;
+  bool occzoomtohighlightedentity;
+  double occdeflection;
+
+  bool whitebackground;
+  int stereo;
+  bool usedispllists;
+  bool drawcoordinatecross;
+  bool drawcolorbar;
+  bool drawnetgenlogo;
+
+  bool use_center_coords;
+  double centerx,centery,centerz;
+
+  
+public:
+  VisualizationParameters();
+};
+extern VisualizationParameters vispar;
+
+#endif
diff --git a/Netgen/libsrc/visualization/vscsg.cpp b/Netgen/libsrc/visualization/vscsg.cpp
index 42891f0a63..d17c0c38e2 100644
--- a/Netgen/libsrc/visualization/vscsg.cpp
+++ b/Netgen/libsrc/visualization/vscsg.cpp
@@ -16,7 +16,7 @@ namespace netgen
 
 
 
-extern CSGeometry * geometry;
+extern AutoPtr<CSGeometry> geometry;
 
 
 VisualSceneGeometry :: VisualSceneGeometry ()
diff --git a/Netgen/libsrc/visualization/vsmesh.cpp b/Netgen/libsrc/visualization/vsmesh.cpp
index f88cb55284..c53e5223be 100644
--- a/Netgen/libsrc/visualization/vsmesh.cpp
+++ b/Netgen/libsrc/visualization/vsmesh.cpp
@@ -66,28 +66,16 @@ namespace netgen
     ;
   }
 
-
-
   
-  ARRAY<Point3d> drawel;
+  // ARRAY<Point3d> drawel;
+
   void VisualSceneMesh :: DrawScene ()
   {
-    /*
-      if (multithread.running)
-      {
-      VisualScene::DrawScene();      
-      return;
-      }
-    */
-    int i;
-    
-    
     if (!mesh) 
       {
 	VisualScene::DrawScene();
 	return;
       }
-
     
     lock = NULL;
     
@@ -137,14 +125,19 @@ namespace netgen
   
     glPolygonMode (GL_FRONT_AND_BACK, GL_FILL);
 
-    glPolygonOffset (1,1);
+    // glPolygonOffset (1,10);
+    glPolygonOffset (2,2);
     glEnable (GL_POLYGON_OFFSET_FILL);
 
     SetClippingPlane ();
 
     if (vispar.drawfilledtrigs)
       {
-	BuildFilledList ();
+	if (filledtimestamp < mesh->GetTimeStamp () ||
+	    filledtimestamp < selecttimestamp)
+	  {
+	    BuildFilledList ();
+	  }
 	glCallList (filledlist);
       }
 
@@ -178,8 +171,6 @@ namespace netgen
 	glCallList (hexlist);
       }
 
-
-
     if (vispar.drawtets)
       {
 	BuildTetList ();
@@ -188,11 +179,13 @@ namespace netgen
 	glLineWidth (1.0f);
 	glCallList (tetlist);
       }
+
     if (vispar.drawdomainsurf)
       {
 	BuildDomainSurfList();
 	glCallList (domainsurflist);
       }
+
     glDisable (GL_POLYGON_OFFSET_FILL);
   
     // draw lines
@@ -210,38 +203,19 @@ namespace netgen
   
     if (vispar.drawoutline)
       {
-	glPolygonOffset (1, -1);
+	glPolygonOffset (1, 1);
 	glEnable (GL_POLYGON_OFFSET_LINE);
 
-	BuildLineList ();
+	if (linetimestamp < mesh->GetTimeStamp ())
+	  BuildLineList ();
+
 	glCallList (linelist);
-      
 	glDisable (GL_POLYGON_OFFSET_LINE);
       }
 
-    /*
-      // not drawing subdivision mesh
-
-    if (vispar.drawtets)
-      {
-	glCallList (tetlist);
-      }
-
-    if (vispar.drawprisms)
-      {
-	glCallList (prismlist);
-      }
-
-    if (vispar.drawpyramids)
-      {
-	glCallList (pyramidlist);
-      }
-
-    */
-
     if (vispar.drawidentified)
       {
-	glPolygonOffset (-1, -1);
+	glPolygonOffset (1, -1);
 	glEnable (GL_POLYGON_OFFSET_LINE);
 	glCallList (identifiedlist);
 	glDisable (GL_POLYGON_OFFSET_LINE);
@@ -262,17 +236,17 @@ namespace netgen
 	GLfloat matcolsingedge[] = { 1, 0, 1, 1 };
 
 	glEnable (GL_POLYGON_OFFSET_LINE);
-	glPolygonOffset (-1, -1);
+	glPolygonOffset (1, -1);
 	glLineWidth (2);
       
 
-	for (i = 1; i <= mesh->GetNSeg(); i++)
+	for (int i = 1; i <= mesh->GetNSeg(); i++)
 	  {
 	    const Segment & seg = mesh->LineSegment(i);
 	    const Point3d & p1 = (*mesh)[seg.p1];
 	    const Point3d & p2 = (*mesh)[seg.p2];
 
-	    if (seg.singedge)
+	    if (seg.singedge_left || seg.singedge_right)
 	      glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, 
 			    matcolsingedge);
 	    else
@@ -293,15 +267,16 @@ namespace netgen
 	      Point<3> x;
 	      glBegin (GL_LINE_STRIP);
 
-	      for (j = 0; j <= hoplotn; j++) {
-		mesh->GetCurvedElements().CalcSegmentTransformation ((double) j/hoplotn, i-1, x);
-		glVertex3d (x(0), x(1), x(2));
-	      }
-
+	      for (int j = 0; j <= hoplotn; j++) 
+		{
+		  mesh->GetCurvedElements().CalcSegmentTransformation ((double) j/hoplotn, i-1, x);
+		  glVertex3d (x(0), x(1), x(2));
+		}
+	      
 	      glEnd();
-             
+	      
 	    } else {
-
+	      
 	      glBegin (GL_LINES);
 	      glVertex3f (p1.X(), p1.Y(), p1.Z());
 	      glVertex3f (p2.X(), p2.Y(), p2.Z());
@@ -363,17 +338,23 @@ namespace netgen
 	
     ARRAY<Element2d> faces;
 
-    if (mesh->GetTimeStamp() > vstimestamp || zoomall)
+    int meshtimestamp = mesh->GetTimeStamp();
+    if (meshtimestamp > vstimestamp || zoomall)
       {
 	mesh->GetBox (pmin, pmax, SURFACEPOINT);
+	
 
 	if (selpoint >= 1 && zoomall == 2)
 	  center = mesh->Point (selpoint);
+	else if (vispar.use_center_coords && zoomall == 2)
+	  {
+	    center.X() = vispar.centerx; center.Y() = vispar.centery; center.Z() = vispar.centerz;
+	  }
 	else if (vispar.centerpoint >= 1 && zoomall == 2)
 	  center = mesh->Point (vispar.centerpoint);
 	else
 	  center = Center (pmin, pmax);
-      
+	        
 	rad = 0.5 * Dist (pmin, pmax);
       
       
@@ -384,7 +365,6 @@ namespace netgen
 	    CalcTransformationMatrices();
 	    oldrad = rad;
 	  }
-	vstimestamp = mesh->GetTimeStamp();
       }
 
     glEnable (GL_NORMALIZE);
@@ -400,21 +380,25 @@ namespace netgen
 	glDeleteLists (badellist, 1);
 	badellist = 0;
       }
-    if (prismlist)
+    /*
+      if (prismlist)
       {
-	glDeleteLists (prismlist, 1);
-	prismlist = 0;
+      glDeleteLists (prismlist, 1);
+      prismlist = 0;
       }
-    if (pyramidlist)
+
+      if (pyramidlist)
       {
-	glDeleteLists (pyramidlist, 1);
-	pyramidlist = 0;
+      glDeleteLists (pyramidlist, 1);
+      pyramidlist = 0;
       }
-    if (hexlist)
+
+      if (hexlist)
       {
-	glDeleteLists (hexlist, 1);
-	hexlist = 0;
+      glDeleteLists (hexlist, 1);
+      hexlist = 0;
       }
+    */
     if (identifiedlist)
       {
 	glDeleteLists (identifiedlist, 1);
@@ -743,7 +727,7 @@ namespace netgen
 		}
 	      default:
 		PrintSysError ("Cannot draw surface element of type ", 
-				 int(el.GetType()));
+			       int(el.GetType()));
 	      }
 	  }
 	glLoadName (0);  
@@ -791,6 +775,8 @@ namespace netgen
 
 	glEndList ();
       }
+
+    vstimestamp = meshtimestamp;
   }
 
 
@@ -798,16 +784,9 @@ namespace netgen
 
   void VisualSceneMesh :: BuildFilledList()
   {
-    clock_t starttime, endtime;
-    starttime = clock();
-
-    int i, j;
-    SurfaceElementIndex sei;
+    // clock_t starttime, endtime;
+    // starttime = clock();
 
-    if (filledtimestamp > mesh->GetTimeStamp () &&
-	filledtimestamp > selecttimestamp)
-      return;
-  
     if (!lock)
       {
 	lock = new NgLock (mesh->Mutex());
@@ -819,13 +798,12 @@ namespace netgen
     if (filledlist)
       glDeleteLists (filledlist, 1);
 
-
-    int checkvicinity = (stlgeometry != NULL) && stldoctor.showvicinity;
-
     filledlist = glGenLists (1);
     glNewList (filledlist, GL_COMPILE);
 
 
+    bool checkvicinity = (stlgeometry != NULL) && stldoctor.showvicinity;
+
     glEnable (GL_NORMALIZE);
       
     glLineWidth (1.0f);
@@ -838,15 +816,13 @@ namespace netgen
 	locms.SetSize (mesh->GetNP());
 	double maxh = -1;
 	double minh = 1e99;
-	for (i = 1; i <= locms.Size(); i++)
+	for (int i = 1; i <= locms.Size(); i++)
 	  {
 	    Point3d p = mesh->Point(i);
 	    locms.Elem(i) = mesh->GetH (p);
 	    if (locms.Elem(i) > maxh) maxh = locms.Elem(i);
 	    if (locms.Elem(i) < minh) minh = locms.Elem(i);
 	  }
-	//	minh = locms.Min();
-	//	maxh = locms.Max();
 	if (!locms.Size())
 	  { minh = 1; maxh = 10; }
       }
@@ -856,6 +832,10 @@ namespace netgen
 
     GLfloat matcol[] = { 0, 1, 0, 1 };
     GLfloat matcolsel[] = { 1, 0, 0, 1 };
+
+    CurvedElements & curv = mesh->GetCurvedElements();
+    int hoplotn = 1 << vispar.subdivisions; 
+
 	
     for (int col = 1; col <= 2; col++)
       {
@@ -865,17 +845,16 @@ namespace netgen
 	  glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, matcol);
 
 
-	for (sei = 0; sei < mesh->GetNSE(); sei++)
+	for (SurfaceElementIndex sei = 0; sei < mesh->GetNSE(); sei++)
 	  {
 	    const Element2d & el = (*mesh)[sei];
-
+	    
 	    bool drawel = !el.IsDeleted();
 
-	    for (j = 0; j < el.GetNP(); j++)
-	      {
-		if (checkvicinity && !stlgeometry->Vicinity(el.GeomInfoPi(j+1).trignum))
+	    if (checkvicinity)
+	      for (int j = 0; j < el.GetNP(); j++)
+		if (!stlgeometry->Vicinity(el.GeomInfoPi(j+1).trignum))
 		  drawel = 0;
-	      }
 
 	    if (!drawel)
 	      continue;
@@ -892,12 +871,8 @@ namespace netgen
 	      {
 	      case TRIG:
 		{
-		  CurvedElements & curv = mesh->GetCurvedElements();
 		  if (curv.IsHighOrder() && curv.IsSurfaceElementCurved(sei))
 		    {
-		      int hoplotn = 1 << vispar.subdivisions; 
-		      // int hoplotn = curv.GetNVisualSubsecs();
-
 		      Point<2> xr[3];
 		      Point<3> xg;
 		      Vec<3> dx, dy, n;
@@ -913,14 +888,15 @@ namespace netgen
 				  xr[0](0) = (double)    i/hoplotn; xr[0](1) = (double)    j/hoplotn;
 				  xr[1](0) = (double)(i+1)/hoplotn; xr[1](1) = (double)    j/hoplotn;
 				  xr[2](0) = (double)    i/hoplotn; xr[2](1) = (double)(j+1)/hoplotn;
-				} else
-				  {
-				    if (j == hoplotn-i-1) continue;
-				    xr[0](0) = (double)(i+1)/hoplotn; xr[0](1) = (double)    j/hoplotn;
-				    xr[1](0) = (double)(i+1)/hoplotn; xr[1](1) = (double)(j+1)/hoplotn;
-				    xr[2](0) = (double)    i/hoplotn; xr[2](1) = (double)(j+1)/hoplotn;
-				  };
-
+				} 
+			      else
+				{
+				  if (j == hoplotn-i-1) continue;
+				  xr[0](0) = (double)(i+1)/hoplotn; xr[0](1) = (double)    j/hoplotn;
+				  xr[1](0) = (double)(i+1)/hoplotn; xr[1](1) = (double)(j+1)/hoplotn;
+				  xr[2](0) = (double)    i/hoplotn; xr[2](1) = (double)(j+1)/hoplotn;
+				};
+			      
 			      for (int l=0; l<3; l++)
 				{
 				  Mat<3,2> dxdxi;
@@ -937,50 +913,47 @@ namespace netgen
 				  glVertex3d (xg(0), xg(1), xg(2));
 				}
 			    }
-  
+		      
 		      glEnd();
+		    } 
+		  else // not high order
+		    {
+		      glBegin (GL_TRIANGLES);
+		      
+		      const Point<3> & lp0 = (*mesh) [el[0]];
+		      const Point<3> & lp1 = (*mesh) [el[1]];
+		      const Point<3> & lp2 = (*mesh) [el[2]];
+		      
+		      Vec<3> n = Cross (lp1-lp0, lp2-lp0);
+		      glNormal3dv (n);
+
+		      if (vispar.colormeshsize)
+			{
+			  SetOpenGlColor  (locms.Get(el[0]), minh, maxh, 1);
+			  glVertex3dv (lp0);
+			  SetOpenGlColor  (locms.Get(el[1]), minh, maxh, 1);
+			  glVertex3dv (lp1);
+			  SetOpenGlColor  (locms.Get(el[2]), minh, maxh, 1);
+			  glVertex3dv (lp2);
+			}
+		      else
+			{
+			  glVertex3dv (lp0);
+			  glVertex3dv (lp1);
+			  glVertex3dv (lp2);
+			}
 
-
-		    } else // not high order
-		      {
-			  
-			glBegin (GL_TRIANGLES);
-
-			Point<3> lp0 = (*mesh) [el.PNum(1)];
-			Point<3> lp1 = (*mesh) [el.PNum(2)];
-			Point<3> lp2 = (*mesh) [el.PNum(3)];
-                  
-			Vec<3> n = Cross (lp1-lp0, lp2-lp0);
-			n.Normalize();
-                  
-			glNormal3d (n(0), n(1), n(2));
-
-			    
-			if (vispar.colormeshsize)
-			  SetOpenGlColor  (locms.Get(el.PNum(1)), minh, maxh, 1);
-			glVertex3d (lp0(0), lp0(1), lp0(2));
-			if (vispar.colormeshsize)
-			  SetOpenGlColor  (locms.Get(el.PNum(2)), minh, maxh, 1);
-			glVertex3d (lp1(0), lp1(1), lp1(2));
-			if (vispar.colormeshsize)
-			  SetOpenGlColor  (locms.Get(el.PNum(3)), minh, maxh, 1);
-			glVertex3d (lp2(0), lp2(1), lp2(2));
-                  
-			glEnd();
-		      }
-
+		      glEnd();
+		    }
+		  
 		  break;
-
 		}
 	      case QUAD:
 		{
 		  // cout << "BuildFilledList: QUAD" << endl;
-		  CurvedElements & curv = mesh->GetCurvedElements();
+		  // CurvedElements & curv = mesh->GetCurvedElements();
 		  if (curv.IsHighOrder() && curv.IsSurfaceElementCurved(sei))
 		    {
-		      int hoplotn = 1 << vispar.subdivisions; 
-		      // int hoplotn = curv.GetNVisualSubsecs();
-
 		      Point<2> xr[4];
 		      Point<3> xg;
 		      Vec<3> dx, dy, n;
@@ -1015,31 +988,31 @@ namespace netgen
 			  }
                     
 		      glEnd();
+		    } 
 
+		  else // not high order
 
-		    } else // not high order
-		      {
-                  
-			glBegin (GL_QUADS);
-
-			const Point3d & lp1 = mesh->Point (el.PNum(1));
-			const Point3d & lp2 = mesh->Point (el.PNum(2));
-			const Point3d & lp3 = mesh->Point (el.PNum(4));
-			const Point3d & lp4 = mesh->Point (el.PNum(3));
-			Vec3d n = Cross (Vec3d (lp1, lp2),
-					 Vec3d (lp1, Center (lp3, lp4)));
-			n /= (n.Length()+1e-12);
-			glNormal3d (n.X(), n.Y(), n.Z());
-			glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
-			glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
-			glVertex3d (lp4.X(), lp4.Y(), lp4.Z());
-			glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
-
-			glEnd ();
-                
-		      }
+		    {
+		      glBegin (GL_QUADS);
+		      
+		      const Point<3> & lp1 = mesh->Point (el.PNum(1));
+		      const Point<3> & lp2 = mesh->Point (el.PNum(2));
+		      const Point<3> & lp3 = mesh->Point (el.PNum(4));
+		      const Point<3> & lp4 = mesh->Point (el.PNum(3));
+
+		      Vec<3> n = Cross (lp2-lp1,  Center (lp3, lp4)-lp1);
+		      glNormal3dv (n);
+
+		      glVertex3dv (lp1);
+		      glVertex3dv (lp2);
+		      glVertex3dv (lp4);
+		      glVertex3dv (lp3);
+		      
+		      glEnd ();
+		    }
 		  break;
 		}
+
 	      case TRIG6:
 		{
 		  glBegin (GL_TRIANGLES);
@@ -1050,14 +1023,14 @@ namespace netgen
 		    { 3, 5, 4 },
 		    { 4, 5, 6 } };
 		
-		  for (j = 0; j < 4; j++)
+		  for (int j = 0; j < 4; j++)
 		    {
 		      Point3d & lp1 = mesh->Point (el.PNum(trigs[j][0]));
 		      Point3d & lp2 = mesh->Point (el.PNum(trigs[j][1]));
 		      Point3d & lp3 = mesh->Point (el.PNum(trigs[j][2]));
 		      Vec3d n = Cross (Vec3d (lp1, lp2), Vec3d (lp1, lp3));
-		      n /= (n.Length() + 1e-12);
 		      glNormal3dv (&n.X());
+
 		      glVertex3dv (&lp1.X());
 		      glVertex3dv (&lp2.X());
 		      glVertex3dv (&lp3.X());
@@ -1065,6 +1038,7 @@ namespace netgen
 		  glEnd();
 		  break;
 		}
+
 	      case QUAD6:
 		{
 		  glBegin (GL_QUADS);
@@ -1072,7 +1046,7 @@ namespace netgen
 		    { 1, 5, 6, 4 },
 		    { 5, 2, 3, 6 } };
 		
-		  for (j = 0; j < 2; j++)
+		  for (int j = 0; j < 2; j++)
 		    {
 		      Point3d & lp1 = mesh->Point (el.PNum(quads[j][0]));
 		      Point3d & lp2 = mesh->Point (el.PNum(quads[j][1]));
@@ -1089,30 +1063,64 @@ namespace netgen
 		  glEnd();
 		  break;
 		}
+
+	      case QUAD8:
+		{
+		  glBegin (GL_TRIANGLES);
+		  static int boundary[] = 
+		    { 1, 5, 2, 8, 3, 6, 4, 7, 1 };
+		
+		  Point3d c(0,0,0);
+		  for (int j = 0; j < 4; j++)
+		    {
+		      Point3d & hp = mesh->Point (el[j]);
+		      c.X() -= 0.25 * hp.X();
+		      c.Y() -= 0.25 * hp.Y();
+		      c.Z() -= 0.25 * hp.Z();
+		    }
+		  for (int j = 4; j < 8; j++)
+		    {
+		      Point3d & hp = mesh->Point (el[j]);
+		      c.X() += 0.5 * hp.X();
+		      c.Y() += 0.5 * hp.Y();
+		      c.Z() += 0.5 * hp.Z();
+		    }
+
+		  for (int j = 0; j < 8; j++)
+		    {
+		      Point3d & lp1 = mesh->Point (el.PNum(boundary[j]));
+		      Point3d & lp2 = mesh->Point (el.PNum(boundary[j+1]));
+
+		      Vec3d n = Cross (Vec3d (c, lp1), Vec3d (c, lp2));
+		      n /= (n.Length() + 1e-12);
+		      glNormal3dv (&n.X());
+		      glVertex3dv (&lp1.X());
+		      glVertex3dv (&lp2.X());
+		      glVertex3dv (&c.X());
+		    }
+		  glEnd();
+		  break;
+		}
+
+
 	      default:
 		PrintSysError ("Cannot draw (2) surface element of type ", 
-				 int(el.GetType()));
+			       int(el.GetType()));
 	      }
 	  }
       }
     glLoadName (0);
     glEndList ();
 
-    endtime = clock();
-
+    // endtime = clock();
     // cout << "BuildFillList time = " << double(endtime - starttime)/CLOCKS_PER_SEC << endl;
   }
 
 
   void VisualSceneMesh :: BuildLineList()
   {
-    int j;
     SurfaceElementIndex sei;
 
-    if (linetimestamp > mesh->GetTimeStamp () &&
-	linetimestamp > selecttimestamp)
-      return;
-
     if (!lock)
       {
 	lock = new NgLock (mesh->Mutex());
@@ -1122,241 +1130,76 @@ namespace netgen
     linetimestamp = NextTimeStamp();
 
 
-    int checkvicinity = (stlgeometry != NULL) && stldoctor.showvicinity;
+    bool checkvicinity = (stlgeometry != NULL) && stldoctor.showvicinity;
 
     if (linelist)
       glDeleteLists (linelist, 1);
-
   
     linelist = glGenLists (1);
     glNewList (linelist, GL_COMPILE);
 
 
     glLineWidth (1.0f);
+
+    int hoplotn = 1 << vispar.subdivisions; 
   
     for (sei = 0; sei < mesh->GetNSE(); sei++)
       {
 	const Element2d & el = (*mesh)[sei];
       
 	bool drawel = !el.IsDeleted();
-	for (j = 0; j < el.GetNP(); j++)
-	  {
-	    if (checkvicinity && !stlgeometry->Vicinity(el.GeomInfoPi(j+1).trignum))
+	if (checkvicinity)
+	  for (int j = 0; j < el.GetNP(); j++)
+	    if (!stlgeometry->Vicinity(el.GeomInfoPi(j+1).trignum))
 	      drawel = 0;
-	  }
 
 	if (!drawel)
 	  continue;
 
 	switch (el.GetType())
 	  {
-	    /*
-	      case TRIG:
-	      {
-	      CurvedElements & curv = mesh->GetCurvedElements();
-	      if (curv.IsHighOrder() && curv.IsSurfaceElementCurved(sei))
-	      {
-	      const MeshTopology & top = mesh->GetTopology();
-                  
-	      int hoplotn = 1 << vispar.subdivisions; 	      
-	      // int hoplotn = curv.GetNVisualSubsecs();
-		  
-	      Point<2> xr[3];
-	      Point<3> xg;
-	      Vec<3> dx, dy, n;
-		  
-	      glBegin (GL_TRIANGLES);
-
-	      for (int i = 0; i < hoplotn; i++)
-	      for (int j = 0; j < hoplotn-i; j++)
-	      for (int k = 0; k < 2; k++)
-	      {
-	      if (k == 0)
-	      {
-	      xr[0](0) = (double)    i/hoplotn; xr[0](1) = (double)    j/hoplotn;
-	      xr[1](0) = (double)(i+1)/hoplotn; xr[1](1) = (double)    j/hoplotn;
-	      xr[2](0) = (double)    i/hoplotn; xr[2](1) = (double)(j+1)/hoplotn;
-	      } else
-	      {
-	      if (j == hoplotn-i-1) continue;
-	      xr[0](0) = (double)(i+1)/hoplotn; xr[0](1) = (double)    j/hoplotn;
-	      xr[1](0) = (double)(i+1)/hoplotn; xr[1](1) = (double)(j+1)/hoplotn;
-	      xr[2](0) = (double)    i/hoplotn; xr[2](1) = (double)(j+1)/hoplotn;
-	      };
-
-	      for (int l=0; l<3; l++)
-	      {
-	      Mat<3,2> dxdxi;
-
-	      curv.CalcSurfaceTransformation (xr[l], sei, xg, dxdxi);
-	      for (int i = 0; i < 3; i++)
-	      {
-	      dx(i) = dxdxi(i,0);
-	      dy(i) = dxdxi(i,1);
-	      }
-	      n = Cross (dx, dy);
-	      n.Normalize();
-	      glNormal3d (n(0), n(1), n(2));
-	      glVertex3d (xg(0), xg(1), xg(2));
-	      }
-	      }
-  
-	      glEnd();
-
-
-	      } else // not high order
-	      {
-	      glBegin (GL_TRIANGLES);
-
-	      Point<3> lp0 = mesh->Point (el.PNum(1));
-	      Point<3> lp1 = mesh->Point (el.PNum(2));
-	      Point<3> lp2 = mesh->Point (el.PNum(3));
-                  
-	      Vec<3> n = Cross (lp1-lp0, lp2-lp0);
-	      n.Normalize();
-                  
-	      glNormal3d (n(0), n(1), n(2));
-	      glVertex3d (lp0(0), lp0(1), lp0(2));
-	      glVertex3d (lp1(0), lp1(1), lp1(2));
-	      glVertex3d (lp2(0), lp2(1), lp2(2));
-                  
-	      glEnd();
-	      }
-
-	      break;
-
-	      }
-
-	      case QUAD:
-	      {
-	      // cout << "BuildFilledList: QUAD" << endl;
-	      CurvedElements & curv = mesh->GetCurvedElements();
-	      if (curv.IsHighOrder() && curv.IsSurfaceElementCurved(sei))
-	      {
-	      const MeshTopology & top = mesh->GetTopology();
-
-	      int hoplotn = 1 << vispar.subdivisions; 
-	      // int hoplotn = curv.GetNVisualSubsecs();
-
-	      Point<2> xr[4];
-	      Point<3> xg;
-	      Vec<3> dx, dy, n;
-
-	      glBegin (GL_QUADS);
-
-	      for (int i = 0; i < hoplotn; i++)
-	      for (int j = 0; j < hoplotn; j++)
-	      {
-	      xr[0](0) = (double)    i/hoplotn; xr[0](1) = (double)    j/hoplotn;
-	      xr[1](0) = (double)(i+1)/hoplotn; xr[1](1) = (double)    j/hoplotn;
-	      xr[2](0) = (double)(i+1)/hoplotn; xr[2](1) = (double)(j+1)/hoplotn;
-	      xr[3](0) = (double)    i/hoplotn; xr[3](1) = (double)(j+1)/hoplotn;
-
-	      for (int l=0; l<4; l++)
-	      {
-	      Mat<3,2> dxdxi;
-			    
-	      curv.CalcSurfaceTransformation (xr[l], sei, xg, dxdxi);
-	      for (int i = 0; i < 3; i++)
-	      {
-	      dx(i) = dxdxi(i,0);
-	      dy(i) = dxdxi(i,1);
-	      }
-			    
-	      n = Cross (dx, dy);
-	      n.Normalize();
-	      glNormal3d (n(0), n(1), n(2));
-	      glVertex3d (xg(0), xg(1), xg(2));
-	      }
-
-	      }
-                    
-	      glEnd();
-
-
-	      } else // not high order
-	      {
-		    
-	      glBegin (GL_QUADS);
-
-	      const Point3d & lp1 = mesh->Point (el.PNum(1));
-	      const Point3d & lp2 = mesh->Point (el.PNum(2));
-	      const Point3d & lp3 = mesh->Point (el.PNum(4));
-	      const Point3d & lp4 = mesh->Point (el.PNum(3));
-	      Vec3d n = Cross (Vec3d (lp1, lp2),
-	      Vec3d (lp1, Center (lp3, lp4)));
-	      n /= (n.Length()+1e-12);
-	      glNormal3d (n.X(), n.Y(), n.Z());
-	      glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
-	      glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
-	      glVertex3d (lp4.X(), lp4.Y(), lp4.Z());
-	      glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
-
-	      glEnd ();
-                
-	      }
-	      break;
-	      }
-	    */
-
 	  case TRIG:
 	    {
 	      CurvedElements & curv = mesh->GetCurvedElements();
 	      if (curv.IsHighOrder() && curv.IsSurfaceElementCurved(sei))
 		{
-		  Point<2> xr;
 		  Point<3> xg;
-		  Vec<3> n;
-                
-		  int hoplotn = 1 << vispar.subdivisions; 
-		  // int hoplotn = curv.GetNVisualSubsecs();
-
-		  glBegin (GL_LINE_STRIP);
+		  glBegin (GL_LINE_LOOP);
 
-		  for (int side = 0; side < 3; side++)
+		  for (int i = 0; i < hoplotn; i++)
 		    {
-		      for (int i = 0; i <= hoplotn; i++)
-			{
-			  switch (side)
-			    {
-			    case 0:
-			      xr(0) = (double) i/hoplotn;
-			      xr(1) = 0.;
-			      break;
-			    case 1:
-			      xr(0) = (double) (hoplotn-i)/hoplotn;
-			      xr(1) = (double)    i/hoplotn;
-			      break;
-			    case 2:
-			      xr(0) = 0.;
-			      xr(1) = (double) (hoplotn-i)/hoplotn;
-			      break;
-			    }
-
-			  Mat<3,2> dxdxi;
-                        
-			  curv.CalcSurfaceTransformation (xr, sei, xg, dxdxi);
-			  glVertex3d (xg(0), xg(1), xg(2));
-
-			}
+		      Point<2> xr (double(i) / hoplotn, 0);
+		      curv.CalcSurfaceTransformation (xr, sei, xg);
+		      glVertex3dv (xg);
+		    }
+		  for (int i = 0; i < hoplotn; i++)
+		    {
+		      Point<2> xr (double(hoplotn-i) / hoplotn, double(i)/hoplotn);
+		      curv.CalcSurfaceTransformation (xr, sei, xg);
+		      glVertex3dv (xg);
+		    }
+		  for (int i = 0; i < hoplotn; i++)
+		    {
+		      Point<2> xr (0, double(hoplotn-i) / hoplotn);
+		      curv.CalcSurfaceTransformation (xr, sei, xg);
+		      glVertex3dv (xg);
+		    }
 
-		    }                       
 		  glEnd();
-
-		} else {
-
+		} 
+	      else 
+		{
 		  glBegin (GL_TRIANGLES);
 
-		  Point<3> lp0 = (*mesh) [el.PNum(1)];
-		  Point<3> lp1 = (*mesh) [el.PNum(2)];
-		  Point<3> lp2 = (*mesh) [el.PNum(3)];
+		  const Point<3> & lp0 = (*mesh) [el[0]];
+		  const Point<3> & lp1 = (*mesh) [el[1]];
+		  const Point<3> & lp2 = (*mesh) [el[2]];
 
 		  glVertex3dv (lp0);
-		  glVertex3d (lp1(0), lp1(1), lp1(2));
-		  glVertex3d (lp2(0), lp2(1), lp2(2));
+		  glVertex3dv (lp1);
+		  glVertex3dv (lp2);
 
 		  glEnd();
-
 		}
 
 	      break;
@@ -1371,9 +1214,6 @@ namespace netgen
 		  Point<2> xr;
 		  Point<3> xg;
 
-		  int hoplotn = 1 << vispar.subdivisions; 
-		  // int hoplotn = curv.GetNVisualSubsecs();
-
 		  glBegin (GL_LINE_STRIP);
 
 		  for (int side = 0; side < 4; side++)
@@ -1400,9 +1240,7 @@ namespace netgen
 			      break;
 			    }
 
-			  Mat<3,2> dxdxi;
-
-			  curv.CalcSurfaceTransformation (xr, sei, xg, dxdxi);
+			  curv.CalcSurfaceTransformation (xr, sei, xg);
 			  glVertex3d (xg(0), xg(1), xg(2));
 
 			}
@@ -1420,7 +1258,6 @@ namespace netgen
 		  const Point3d & lp4 = mesh->Point (el.PNum(3));
 		  Vec3d n = Cross (Vec3d (lp1, lp2),
 				   Vec3d (lp1, Center (lp3, lp4)));
-		  n /= (n.Length() + 1e-12);
 		  glNormal3d (n.X(), n.Y(), n.Z());
 		  glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
 		  glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
@@ -1442,7 +1279,7 @@ namespace netgen
 		{ 2, 4 }, { 3, 4 } };
 	    
 	      glBegin (GL_LINES);
-	      for (j = 0; j < 6; j++)
+	      for (int j = 0; j < 6; j++)
 		{
 		  const Point3d & lp1 = mesh->Point (el.PNum(lines[j][0]));
 		  const Point3d & lp2 = mesh->Point (el.PNum(lines[j][1]));
@@ -1464,7 +1301,28 @@ namespace netgen
 	    
 	      glBegin (GL_LINES);
 	    
-	      for (j = 0; j < 6; j++)
+	      for (int j = 0; j < 6; j++)
+		{
+		  const Point3d & lp1 = mesh->Point (el.PNum(lines[j][0]));
+		  const Point3d & lp2 = mesh->Point (el.PNum(lines[j][1]));
+		
+		  glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
+		  glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
+		}
+	      glEnd ();
+	      break;
+	    }
+
+	  case QUAD8:
+	    {
+	      int lines[8][2] = {
+		{ 1, 5 }, { 2, 5 }, { 3, 6 }, { 4, 6 },
+		{ 1, 7 }, { 4, 7 }, { 2, 8 }, { 3, 8 }
+	      };
+	    
+	      glBegin (GL_LINES);
+	    
+	      for (int j = 0; j < 8; j++)
 		{
 		  const Point3d & lp1 = mesh->Point (el.PNum(lines[j][0]));
 		  const Point3d & lp2 = mesh->Point (el.PNum(lines[j][1]));
@@ -1476,20 +1334,85 @@ namespace netgen
 	      break;
 	    }
 
-	  default:
-	    PrintSysError ("Cannot draw (4) surface element of type ", 
-			     int(el.GetType()));
-	  }
-      }
-    
-    glEndList ();
-  }
+
+
+	  default:
+	    PrintSysError ("Cannot draw (4) surface element of type ", 
+			   int(el.GetType()));
+	  }
+      }
+    
+    glEndList ();
+  }
 
   void VisualSceneMesh :: BuildPointNumberList()
   {
     ;
   }
 
+
+
+
+
+
+  inline long int Fact (int n)
+  {
+    long int res = 1;
+    for (int i = 2; i <= n; i++)
+      res *= i;
+    return res;
+  }
+  inline int Binom (int n, int i)
+  {
+    return Fact(n) / Fact(i) / Fact(n-i);
+  }
+  
+  void ToBernstein (int order, Point<3> * pts, int stride)
+  {
+    static DenseMatrix mat, inv;
+    static Vector vec1, vec2;
+
+    if (mat.Height () != order+1)
+      {
+	mat.SetSize (order+1);
+	inv.SetSize (order+1);
+	vec1.SetSize (order+1);
+	vec2.SetSize (order+1);
+	for (int i = 0; i <= order; i++)
+	  {
+	    double x = double(i) / order;
+	    for (int j = 0; j <= order; j++)
+	      mat(i,j) = Binom (order, j) * pow (x, j) * pow (1-x, order-j);
+	  }
+
+	CalcInverse (mat, inv);
+      }
+
+    for (int i = 0; i < 3; i++)
+      {
+	for (int j = 0; j <= order; j++)
+	  vec1(j) = pts[j*stride](i);
+
+	inv.Mult (vec1, vec2);
+
+	for (int j = 0; j <= order; j++)
+	  pts[j*stride](i) = vec2(j);
+      }
+  }
+
+
+
+
+
+
+
+
+
+
+
+
+
+
   void VisualSceneMesh :: BuildTetList()
   {
 
@@ -1613,6 +1536,9 @@ namespace netgen
 		if (!drawtet) continue;
 
 		int ind = el.GetIndex() % 4;
+		if (vispar.drawmetispartition && (el.GetPartition()!=-1))
+		  ind = el.GetPartition() % 4;
+		(*testout) << "ind = " << ind << endl;
 		glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, tetcols[ind]);
 
 
@@ -1627,455 +1553,560 @@ namespace netgen
 		    Element2d & face = faces.Elem(j+1);
 
 		    if (curv.IsHighOrder() && curv.IsElementCurved(i-1))
-		  {
-		    int hoplotn = 1 << vispar.subdivisions; 
-		    // int hoplotn = curv.GetNVisualSubsecs();
+		      {
+			int hoplotn = 1 << vispar.subdivisions; 
+			// int hoplotn = curv.GetNVisualSubsecs();
 			    
-		    const Point3d * facepoint = MeshTopology :: GetVertices (TET);
-		    const ELEMENT_FACE * elface = MeshTopology :: GetFaces(TET);
+			const Point3d * facepoint = MeshTopology :: GetVertices (TET);
+			const ELEMENT_FACE * elface = MeshTopology :: GetFaces(TET);
                     
-		    Vec<3> x0,x1,d0,d1;
-		    Point<3> xg;
-		    x0 = facepoint[face.PNum(3)-1] - facepoint[face.PNum(1)-1];
-		    x1 = facepoint[face.PNum(2)-1] - facepoint[face.PNum(1)-1];
-		    x0.Normalize();
-		    x1.Normalize();
-
-		    for (int m0 = 0; m0 < hoplotn; m0++)
-		      for (int m1 = 0; m1 < hoplotn-m0; m1++)
-			for (int k = 0; k < 2; k++)
-			  {
-			    Vec<3> dx, dy, dz, n;
-			    Point<4> la[3];
-			    int l;
-			    for (l = 0; l<3; l++) la[l] = Point<4>(0.,0.,0.,0.);
-
-			    if (k == 0)
+			Vec<3> x0,x1,d0,d1;
+			Point<3> xg;
+			x0 = facepoint[face.PNum(3)-1] - facepoint[face.PNum(1)-1];
+			x1 = facepoint[face.PNum(2)-1] - facepoint[face.PNum(1)-1];
+			x0.Normalize();
+			x1.Normalize();
+
+			for (int m0 = 0; m0 < hoplotn; m0++)
+			  for (int m1 = 0; m1 < hoplotn-m0; m1++)
+			    for (int k = 0; k < 2; k++)
 			      {
-				la[0](face.PNum(1)-1) = (m0  )/(double)hoplotn;
-				la[0](face.PNum(2)-1) = (m1  )/(double)hoplotn;
-				la[0](face.PNum(3)-1) = 1-la[0](face.PNum(1)-1)-la[0](face.PNum(2)-1);
+				Vec<3> dx, dy, dz, n;
+				Point<4> la[3];
+				int l;
+				for (l = 0; l<3; l++) la[l] = Point<4>(0.,0.,0.,0.);
 
-				la[1](face.PNum(1)-1) = (m0+1)/(double)hoplotn;
-				la[1](face.PNum(2)-1) = (m1  )/(double)hoplotn;
-				la[1](face.PNum(3)-1) = 1-la[1](face.PNum(1)-1)-la[1](face.PNum(2)-1);
-
-				la[2](face.PNum(1)-1) = (m0  )/(double)hoplotn;
-				la[2](face.PNum(2)-1) = (m1+1)/(double)hoplotn;
-				la[2](face.PNum(3)-1) = 1-la[2](face.PNum(1)-1)-la[2](face.PNum(2)-1);
-			      } else
-				{
-				  if (m1 == hoplotn-m0-1) continue;
-				  la[0](face.PNum(1)-1) = (m0+1)/(double)hoplotn;
-				  la[0](face.PNum(2)-1) = (m1+1)/(double)hoplotn;
-				  la[0](face.PNum(3)-1) = 1-la[0](face.PNum(1)-1)-la[0](face.PNum(2)-1);
-
-				  la[1](face.PNum(1)-1) = (m0  )/(double)hoplotn;
-				  la[1](face.PNum(2)-1) = (m1+1)/(double)hoplotn;
-				  la[1](face.PNum(3)-1) = 1-la[1](face.PNum(1)-1)-la[1](face.PNum(2)-1);
-
-				  la[2](face.PNum(1)-1) = (m0+1)/(double)hoplotn;
-				  la[2](face.PNum(2)-1) = (m1  )/(double)hoplotn;
-				  la[2](face.PNum(3)-1) = 1-la[2](face.PNum(1)-1)-la[2](face.PNum(2)-1);
-				}
+				if (k == 0)
+				  {
+				    la[0](face.PNum(1)-1) = (m0  )/(double)hoplotn;
+				    la[0](face.PNum(2)-1) = (m1  )/(double)hoplotn;
+				    la[0](face.PNum(3)-1) = 1-la[0](face.PNum(1)-1)-la[0](face.PNum(2)-1);
+
+				    la[1](face.PNum(1)-1) = (m0+1)/(double)hoplotn;
+				    la[1](face.PNum(2)-1) = (m1  )/(double)hoplotn;
+				    la[1](face.PNum(3)-1) = 1-la[1](face.PNum(1)-1)-la[1](face.PNum(2)-1);
+
+				    la[2](face.PNum(1)-1) = (m0  )/(double)hoplotn;
+				    la[2](face.PNum(2)-1) = (m1+1)/(double)hoplotn;
+				    la[2](face.PNum(3)-1) = 1-la[2](face.PNum(1)-1)-la[2](face.PNum(2)-1);
+				  } else
+				    {
+				      if (m1 == hoplotn-m0-1) continue;
+				      la[0](face.PNum(1)-1) = (m0+1)/(double)hoplotn;
+				      la[0](face.PNum(2)-1) = (m1+1)/(double)hoplotn;
+				      la[0](face.PNum(3)-1) = 1-la[0](face.PNum(1)-1)-la[0](face.PNum(2)-1);
+
+				      la[1](face.PNum(1)-1) = (m0  )/(double)hoplotn;
+				      la[1](face.PNum(2)-1) = (m1+1)/(double)hoplotn;
+				      la[1](face.PNum(3)-1) = 1-la[1](face.PNum(1)-1)-la[1](face.PNum(2)-1);
+
+				      la[2](face.PNum(1)-1) = (m0+1)/(double)hoplotn;
+				      la[2](face.PNum(2)-1) = (m1  )/(double)hoplotn;
+				      la[2](face.PNum(3)-1) = 1-la[2](face.PNum(1)-1)-la[2](face.PNum(2)-1);
+				    }
 
-			    for (l = 0; l<3; l++)
-			      {
-				Mat<3,3> dxdxi;
-				Point<3> xr( la[l](0), la[l](1), la[l](2) );
-				curv.CalcElementTransformation (xr, i-1, xg, dxdxi);
-				for (int i = 0; i < 3; i++)
+				for (l = 0; l<3; l++)
 				  {
-				    dx(i) = dxdxi(i,0);
-				    dy(i) = dxdxi(i,1);
-				    dz(i) = dxdxi(i,2);
+				    Mat<3,3> dxdxi;
+				    Point<3> xr( la[l](0), la[l](1), la[l](2) );
+				    curv.CalcElementTransformation (xr, i-1, xg, dxdxi);
+				    for (int i = 0; i < 3; i++)
+				      {
+					dx(i) = dxdxi(i,0);
+					dy(i) = dxdxi(i,1);
+					dz(i) = dxdxi(i,2);
+				      }
+
+				    d0 = x0(0)*dx + x0(1)*dy + x0(2)*dz;
+				    d1 = x1(0)*dx + x1(1)*dy + x1(2)*dz;
+				    n = Cross (d0, d1);
+				    glNormal3d ( n(0),  n(1),  n(2));
+				    glVertex3d (xg(0), xg(1), xg(2));
 				  }
-
-				d0 = x0(0)*dx + x0(1)*dy + x0(2)*dz;
-				d1 = x1(0)*dx + x1(1)*dy + x1(2)*dz;
-				n = Cross (d0, d1);
-				glNormal3d ( n(0),  n(1),  n(2));
-				glVertex3d (xg(0), xg(1), xg(2));
 			      }
-			  }
 
-		  } else {
-		    const Point3d & lp1 = mesh->Point (el.PNum(face.PNum(1)));
-		    const Point3d & lp2 = mesh->Point (el.PNum(face.PNum(2)));
-		    const Point3d & lp3 = mesh->Point (el.PNum(face.PNum(3)));
-		    Vec3d n = Cross (Vec3d (lp1, lp3), Vec3d (lp1, lp2));
-		    n /= (n.Length()+1e-12);
-		    glNormal3d (n.X(), n.Y(), n.Z());
-		    glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
-		    glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
-		    glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
+		      } else {
+			const Point3d & lp1 = mesh->Point (el.PNum(face.PNum(1)));
+			const Point3d & lp2 = mesh->Point (el.PNum(face.PNum(2)));
+			const Point3d & lp3 = mesh->Point (el.PNum(face.PNum(3)));
+			Vec3d n = Cross (Vec3d (lp1, lp3), Vec3d (lp1, lp2));
+			n /= (n.Length()+1e-12);
+			glNormal3d (n.X(), n.Y(), n.Z());
+			glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
+			glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
+			glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
+		      }
 		  }
-	      }
 	     
-	    glEnd();
-	  }
+		glEnd();
+	      }
 
+	  }
       }
-  }
-  glEndList ();
+    glEndList ();
 
 
 
 
 #else
 
+    if (tettimestamp > mesh->GetTimeStamp () &&
+	tettimestamp > vispar.clipplanetimestamp )
+      return;
+
+    if (!lock)
+      {
+	lock = new NgLock (mesh->Mutex());
+	lock -> Lock();
+      }
 
-  int i, j, k, l;
-  ARRAY<Element2d> faces;
+    tettimestamp = NextTimeStamp();
 
-  if (tettimestamp > mesh->GetTimeStamp () &&
-      tettimestamp > vispar.clipplanetimestamp )
-    return;
+    if (tetlist)
+      glDeleteLists (tetlist, 1);
 
-  if (!lock)
-  {
-    lock = new NgLock (mesh->Mutex());
-    lock -> Lock();
-  }
 
-  tettimestamp = NextTimeStamp();
+    tetlist = glGenLists (1);
+    glNewList (tetlist, GL_COMPILE);
 
-  if (tetlist)
-    glDeleteLists (tetlist, 1);
 
 
-  tetlist = glGenLists (1);
-  glNewList (tetlist, GL_COMPILE);
+    int i, j, k, l;
+    ARRAY<Element2d> faces;
 
   
-  BitArray shownode(mesh->GetNP());
-  if (vispar.clipenable)
-  {
-    shownode.Clear();
-    for (i = 1; i <= shownode.Size(); i++)
+    BitArray shownode(mesh->GetNP());
+    if (vispar.clipenable)
       {
-	Point3d p = mesh->Point(i);
+	shownode.Clear();
+	for (i = 1; i <= shownode.Size(); i++)
+	  {
+	    Point3d p = mesh->Point(i);
 	  
-	double val =
-	  p.X() * clipplane[0] +
-	  p.Y() * clipplane[1] +
-	  p.Z() * clipplane[2] +
-	  clipplane[3];
+	    double val =
+	      p.X() * clipplane[0] +
+	      p.Y() * clipplane[1] +
+	      p.Z() * clipplane[2] +
+	      clipplane[3];
 	  
-	if (val > 0)
-	  shownode.Set (i);
+	    if (val > 0)
+	      shownode.Set (i);
+	  }
       }
-  }
-  else
-  shownode.Set();
+    else
+      shownode.Set();
 
 
-  static float tetcols[][4] = 
-    {
-      { 1.0f, 1.0f, 0.0f, 1.0f },
-      { 1.0f, 0.0f, 0.0f, 1.0f },
-      { 0.0f, 1.0f, 0.0f, 1.0f },
-      { 0.0f, 0.0f, 1.0f, 1.0f }
-    };
+    static float tetcols[][4] = 
+      {
+	{ 1.0f, 1.0f, 0.0f, 1.0f },
+	{ 1.0f, 0.0f, 0.0f, 1.0f },
+	{ 0.0f, 1.0f, 0.0f, 1.0f },
+	{ 0.0f, 0.0f, 1.0f, 1.0f }
+      };
 
 
-  // ARRAY<int> elfaces;
+    CurvedElements & curv = mesh->GetCurvedElements();
 
-  // const MeshTopology & top = mesh->GetTopology();
-  CurvedElements & curv = mesh->GetCurvedElements();
-    
+    for (ElementIndex ei = 0; ei < mesh->GetNE(); ei++)
+      {
+	i = ei + 1;
 
-  for (i = 1; i <= mesh->GetNE(); i++)
-  {
-    if (vispar.drawtetsdomain > 0 &&
-	vispar.drawtetsdomain != mesh->VolumeElement(i).GetIndex())
-      continue;
+	if (vispar.drawtetsdomain > 0 &&
+	    vispar.drawtetsdomain != mesh->VolumeElement(i).GetIndex())
+	  continue;
 
-    if (mesh->VolumeElement(i).GetType() == TET)
-      {
-	// copy to be thread-safe
-	Element el = mesh->VolumeElement (i);
-	el.GetSurfaceTriangles (faces);
+	const Element & el = (*mesh)[ei];
 
-	if (el.PNum(1))
+	if (el.GetType() == TET && !el.IsDeleted())
 	  {
-	    int drawtet = 1;
-	    for (j = 1; j <= el.GetNP(); j++)
-	      if (!shownode.Test(el.PNum(j)))
+
+	    bool drawtet = 1;
+	    for (j = 0; j < 4; j++)
+	      if (!shownode.Test(el[j]))
 		drawtet = 0;
 	    if (!drawtet) continue;
-
-	    Point3d c;
-	    if (vispar.shrink < 1)
-	      c = Center (Center (mesh->Point (el.PNum(1)),
-				  mesh->Point (el.PNum(2))),
-			  Center (mesh->Point (el.PNum(3)),
-				  mesh->Point (el.PNum(4))));
-
-
-
+	    
+	    
 	    int ind = el.GetIndex() % 4;
+	    if (vispar.drawmetispartition && (el.GetPartition()!=-1))
+	      ind = el.GetPartition() % 4;
 	    glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, tetcols[ind]);
+	    
+	    
+	    if (curv.IsHighOrder() && curv.IsElementCurved(ei))
+	      {
+		const ELEMENT_FACE * faces = MeshTopology :: GetFaces (TET);
+		const Point3d * vertices = MeshTopology :: GetVertices (TET);
+		
+		Point<3> grid[11][11];
+		Point<3> fpts[3];
+		int order = vispar.subdivisions+1;
+		
+		for (int trig = 0; trig < 4; trig++)
+		  {   
+		    for (int j = 0; j < 3; j++)
+		      fpts[j] = vertices[faces[trig][j]-1];
+		    
+		    static Point<3> c(0.25, 0.25, 0.25);
+		    if (vispar.shrink < 1)
+		      for (int j = 0; j < 3; j++)
+			fpts[j] += (1-vispar.shrink) * (c-fpts[j]);
+			
+		    for (int ix = 0; ix <= order; ix++)
+		      for (int iy = 0; iy <= order; iy++)
+			{
+			  double lami[3] = 
+			    { (1-double(ix)/order) * (1-double(iy)/order),
+			      (  double(ix)/order) * (1-double(iy)/order),
+			      double(iy)/order };
+			      
+			  Point<3> xl;
+			  for (int l = 0; l < 3; l++)
+			    xl(l) = lami[0] * fpts[0](l) + lami[1] * fpts[1](l) +
+			      lami[2] * fpts[2](l);
+			      
+			  curv.CalcElementTransformation (xl, i-1, grid[ix][iy]);
+			}
+			
+		    for (int j = 0; j <= order; j++)
+		      ToBernstein (order, &grid[j][0], &grid[0][1]-&grid[0][0]);
+		    for (int j = 0; j <= order; j++)
+		      ToBernstein (order, &grid[0][j], &grid[1][0]-&grid[0][0]);
+			
+		    glMap2d(GL_MAP2_VERTEX_3, 
+			    0.0, 1.0, &grid[0][1](0)-&grid[0][0](0), order+1,
+			    0.0, 1.0, &grid[1][0](0)-&grid[0][0](0), order+1,
+			    &grid[0][0](0));
+		    glEnable(GL_MAP2_VERTEX_3);
+		    glEnable(GL_AUTO_NORMAL); 
+			
+		    glMapGrid2f(8, 0.0, 0.999, 8, 0.0, 1.0);
+		    glEvalMesh2(GL_FILL, 0, 8, 0, 8);
+			
+		    glDisable (GL_AUTO_NORMAL);
+		    glDisable (GL_MAP2_VERTEX_3);
+		  }		    
+	      }
 
 
-	    // top.GetElementFaces (i, elfaces);
-
-	    glBegin (GL_TRIANGLES);
+	    else
 
-	    for (j = 0; j < faces.Size(); j++)
 	      {
-		Element2d & face = faces.Elem(j+1);
+		el.GetSurfaceTriangles (faces);
+
+		Point3d c;
+		if (vispar.shrink < 1)
+		  c = Center (Center (mesh->Point (el.PNum(1)),
+				      mesh->Point (el.PNum(2))),
+			      Center (mesh->Point (el.PNum(3)),
+				      mesh->Point (el.PNum(4))));
+
 
-		if (curv.IsHighOrder() && curv.IsElementCurved(i-1))
+		glBegin (GL_TRIANGLES);
+
+		for (j = 0; j < faces.Size(); j++)
 		  {
-		    int hoplotn = 1 << vispar.subdivisions; 
-		    // int hoplotn = curv.GetNVisualSubsecs();
+		    const Element2d & face = faces[j];
+
+		    /*
+		      if (curv.IsHighOrder() && curv.IsElementCurved(i-1))
+		      {
+		      int hoplotn = 1 << vispar.subdivisions; 
+		      // int hoplotn = curv.GetNVisualSubsecs();
 			    
-		    const Point3d * facepoint = MeshTopology :: GetVertices (TET);
-		    const ELEMENT_FACE * elface = MeshTopology :: GetFaces(TET);
+		      const Point3d * facepoint = MeshTopology :: GetVertices (TET);
+		      const ELEMENT_FACE * elface = MeshTopology :: GetFaces(TET);
                     
-		    Vec<3> x0,x1,d0,d1;
-		    Point<3> xg;
-		    x0 = facepoint[face.PNum(3)-1] - facepoint[face.PNum(1)-1];
-		    x1 = facepoint[face.PNum(2)-1] - facepoint[face.PNum(1)-1];
-		    x0.Normalize();
-		    x1.Normalize();
+		      Vec<3> x0,x1,d0,d1;
+		      Point<3> xg;
+		      x0 = facepoint[face.PNum(3)-1] - facepoint[face.PNum(1)-1];
+		      x1 = facepoint[face.PNum(2)-1] - facepoint[face.PNum(1)-1];
+		      x0.Normalize();
+		      x1.Normalize();
 
-		    for (int m0 = 0; m0 < hoplotn; m0++)
+		      for (int m0 = 0; m0 < hoplotn; m0++)
 		      for (int m1 = 0; m1 < hoplotn-m0; m1++)
-			for (k = 0; k < 2; k++)
-			  {
-			    Vec<3> dx, dy, dz, n;
-			    Point<4> la[3];
+		      for (k = 0; k < 2; k++)
+		      {
+		      Vec<3> dx, dy, dz, n;
+		      Point<4> la[3];
 					
-			    for (l = 0; l<3; l++) la[l] = Point<4>(0.,0.,0.,0.);
+		      for (l = 0; l<3; l++) la[l] = Point<4>(0.,0.,0.,0.);
 
-			    if (k == 0)
-			      {
-				la[0](face.PNum(1)-1) = (m0  )/(double)hoplotn;
-				la[0](face.PNum(2)-1) = (m1  )/(double)hoplotn;
-				la[0](face.PNum(3)-1) = 1-la[0](face.PNum(1)-1)-la[0](face.PNum(2)-1);
+		      if (k == 0)
+		      {
+		      la[0](face.PNum(1)-1) = (m0  )/(double)hoplotn;
+		      la[0](face.PNum(2)-1) = (m1  )/(double)hoplotn;
+		      la[0](face.PNum(3)-1) = 1-la[0](face.PNum(1)-1)-la[0](face.PNum(2)-1);
+
+		      la[1](face.PNum(1)-1) = (m0+1)/(double)hoplotn;
+		      la[1](face.PNum(2)-1) = (m1  )/(double)hoplotn;
+		      la[1](face.PNum(3)-1) = 1-la[1](face.PNum(1)-1)-la[1](face.PNum(2)-1);
+
+		      la[2](face.PNum(1)-1) = (m0  )/(double)hoplotn;
+		      la[2](face.PNum(2)-1) = (m1+1)/(double)hoplotn;
+		      la[2](face.PNum(3)-1) = 1-la[2](face.PNum(1)-1)-la[2](face.PNum(2)-1);
+		      } else
+		      {
+		      if (m1 == hoplotn-m0-1) continue;
+		      la[0](face.PNum(1)-1) = (m0+1)/(double)hoplotn;
+		      la[0](face.PNum(2)-1) = (m1+1)/(double)hoplotn;
+		      la[0](face.PNum(3)-1) = 1-la[0](face.PNum(1)-1)-la[0](face.PNum(2)-1);
+
+		      la[1](face.PNum(1)-1) = (m0  )/(double)hoplotn;
+		      la[1](face.PNum(2)-1) = (m1+1)/(double)hoplotn;
+		      la[1](face.PNum(3)-1) = 1-la[1](face.PNum(1)-1)-la[1](face.PNum(2)-1);
+
+		      la[2](face.PNum(1)-1) = (m0+1)/(double)hoplotn;
+		      la[2](face.PNum(2)-1) = (m1  )/(double)hoplotn;
+		      la[2](face.PNum(3)-1) = 1-la[2](face.PNum(1)-1)-la[2](face.PNum(2)-1);
+		      }
 
-				la[1](face.PNum(1)-1) = (m0+1)/(double)hoplotn;
-				la[1](face.PNum(2)-1) = (m1  )/(double)hoplotn;
-				la[1](face.PNum(3)-1) = 1-la[1](face.PNum(1)-1)-la[1](face.PNum(2)-1);
+		      for (l = 0; l<3; l++)
+		      {
+		      Mat<3,3> dxdxi;
+		      Point<3> xr( la[l](0), la[l](1), la[l](2) );
+		      curv.CalcElementTransformation (xr, i-1, xg, dxdxi);
+		      for (int i = 0; i < 3; i++)
+		      {
+		      dx(i) = dxdxi(i,0);
+		      dy(i) = dxdxi(i,1);
+		      dz(i) = dxdxi(i,2);
+		      }
 
-				la[2](face.PNum(1)-1) = (m0  )/(double)hoplotn;
-				la[2](face.PNum(2)-1) = (m1+1)/(double)hoplotn;
-				la[2](face.PNum(3)-1) = 1-la[2](face.PNum(1)-1)-la[2](face.PNum(2)-1);
-			      } else
-				{
-				  if (m1 == hoplotn-m0-1) continue;
-				  la[0](face.PNum(1)-1) = (m0+1)/(double)hoplotn;
-				  la[0](face.PNum(2)-1) = (m1+1)/(double)hoplotn;
-				  la[0](face.PNum(3)-1) = 1-la[0](face.PNum(1)-1)-la[0](face.PNum(2)-1);
-
-				  la[1](face.PNum(1)-1) = (m0  )/(double)hoplotn;
-				  la[1](face.PNum(2)-1) = (m1+1)/(double)hoplotn;
-				  la[1](face.PNum(3)-1) = 1-la[1](face.PNum(1)-1)-la[1](face.PNum(2)-1);
-
-				  la[2](face.PNum(1)-1) = (m0+1)/(double)hoplotn;
-				  la[2](face.PNum(2)-1) = (m1  )/(double)hoplotn;
-				  la[2](face.PNum(3)-1) = 1-la[2](face.PNum(1)-1)-la[2](face.PNum(2)-1);
-				}
+		      d0 = x0(0)*dx + x0(1)*dy + x0(2)*dz;
+		      d1 = x1(0)*dx + x1(1)*dy + x1(2)*dz;
+		      n = Cross (d0, d1);
+		      glNormal3d ( n(0),  n(1),  n(2));
+		      glVertex3d (xg(0), xg(1), xg(2));
+		      }
+		      }
 
-			    for (l = 0; l<3; l++)
-			      {
-				Mat<3,3> dxdxi;
-				Point<3> xr( la[l](0), la[l](1), la[l](2) );
-				curv.CalcElementTransformation (xr, i-1, xg, dxdxi);
-				for (int i = 0; i < 3; i++)
-				  {
-				    dx(i) = dxdxi(i,0);
-				    dy(i) = dxdxi(i,1);
-				    dz(i) = dxdxi(i,2);
-				  }
+		      } else 
+		    */
+		    {
+		      Point3d lp1 = mesh->Point (el.PNum(face.PNum(1)));
+		      Point3d lp2 = mesh->Point (el.PNum(face.PNum(2)));
+		      Point3d lp3 = mesh->Point (el.PNum(face.PNum(3)));
+		      Vec3d n = Cross (Vec3d (lp1, lp3), Vec3d (lp1, lp2));
+		      n /= (n.Length()+1e-12);
+		      glNormal3d (n.X(), n.Y(), n.Z());
+
+		      if (vispar.shrink < 1)
+			{
+			  lp1 = c + vispar.shrink * (lp1 - c);
+			  lp2 = c + vispar.shrink * (lp2 - c);
+			  lp3 = c + vispar.shrink * (lp3 - c);
+			}
 
-				d0 = x0(0)*dx + x0(1)*dy + x0(2)*dz;
-				d1 = x1(0)*dx + x1(1)*dy + x1(2)*dz;
-				n = Cross (d0, d1);
-				glNormal3d ( n(0),  n(1),  n(2));
-				glVertex3d (xg(0), xg(1), xg(2));
-			      }
-			  }
-
-		  } else {
-		    Point3d lp1 = mesh->Point (el.PNum(face.PNum(1)));
-		    Point3d lp2 = mesh->Point (el.PNum(face.PNum(2)));
-		    Point3d lp3 = mesh->Point (el.PNum(face.PNum(3)));
-		    Vec3d n = Cross (Vec3d (lp1, lp3), Vec3d (lp1, lp2));
-		    n /= (n.Length()+1e-12);
-		    glNormal3d (n.X(), n.Y(), n.Z());
-
-		    if (vispar.shrink < 1)
-		      {
-			lp1 = c + vispar.shrink * (lp1 - c);
-			lp2 = c + vispar.shrink * (lp2 - c);
-			lp3 = c + vispar.shrink * (lp3 - c);
-		      }
-
-		    glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
-		    glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
-		    glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
-		  }
-	      }
-	     
-	    glEnd();
-	  }
-      }
-  }
-  glEndList ();
+		      glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
+		      glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
+		      glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
+		    }
+		  }
+	     
+		glEnd();
+	      }
+	  }
+      }
+    
+    glEndList ();
+    
+#endif
+  }
 
 
 
-#endif
-}
-
 
-void VisualSceneMesh :: BuildPrismList()
-{
-  int i, j;
+  void VisualSceneMesh :: BuildPrismList()
+  {
+    if (prismtimestamp > mesh->GetTimeStamp () &&
+	prismtimestamp > vispar.clipplanetimestamp )
+      return;
 
-  prismlist = glGenLists (1);
-  glNewList (prismlist, GL_COMPILE);
+    if (!lock)
+      {
+	lock = new NgLock (mesh->Mutex());
+	lock -> Lock();
+      }
 
-  static float prismcol[] = { 0.0f, 1.0f, 1.0f, 1.0f };
-  glLineWidth (1.0f);  
+    prismtimestamp = NextTimeStamp();
 
-  ARRAY<Element2d> faces;
 
-  for (i = 1; i <= mesh->GetNE(); i++)
-    {
-      if (mesh->VolumeElement(i).GetType() == PRISM)
-	{
-	  // copy to be thread-safe
-	  Element el = mesh->VolumeElement (i);
-	  el.GetSurfaceTriangles (faces);
 
-	  glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, prismcol);
-	  if (el.PNum(1))
-	    {
-	      CurvedElements & curv = mesh->GetCurvedElements();
-	      if (curv.IsHighOrder() && curv.IsElementCurved(i-1))
-		{
-		  int hoplotn = 1 << vispar.subdivisions; 
-		  // int hoplotn = curv.GetNVisualSubsecs();
+    if (prismlist)
+      glDeleteLists (prismlist, 1);
 
-		  const Point3d * facepoint = MeshTopology :: GetVertices (TRIG);
-		  const ELEMENT_FACE * elface = MeshTopology :: GetFaces(TRIG);
+    prismlist = glGenLists (1);
+    glNewList (prismlist, GL_COMPILE);
 
-		  glBegin (GL_TRIANGLES);
+    static float prismcol[] = { 0.0f, 1.0f, 1.0f, 1.0f };
+    glLineWidth (1.0f);  
 
-		  for (int trig = 0; trig<2; trig++)
-		    {
-                    
-		      Vec<3> x0,x1,d0,d1;
-		      x0 = facepoint[1] - facepoint[2];
-		      x1 = facepoint[0] - facepoint[2];
-		      x0.Normalize();
-		      x1.Normalize();
-		      if (trig == 1) Swap (x0,x1);
+    ARRAY<Element2d> faces;
 
-		      Point<3> xr[3];
-		      Point<3> xg;
-		      Vec<3> dx, dy, dz, n;
+    
+    glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, prismcol);
 
-		      for (int i1 = 0; i1 < hoplotn; i1++)
-			for (int j1 = 0; j1 < hoplotn-i1; j1++)
-			  for (int k = 0; k < 2; k++)
-			    {
-			      if (k == 0)
-				{
-				  xr[0](0) = (double)    i1/hoplotn; xr[0](1) = (double)    j1/hoplotn;
-				  xr[1](0) = (double)(i1+1)/hoplotn; xr[1](1) = (double)    j1/hoplotn;
-				  xr[2](0) = (double)    i1/hoplotn; xr[2](1) = (double)(j1+1)/hoplotn;
-				} else
-				  {
-				    if (j1 == hoplotn-i1-1) continue;
-				    xr[0](0) = (double)(i1+1)/hoplotn; xr[0](1) = (double)    j1/hoplotn;
-				    xr[1](0) = (double)(i1+1)/hoplotn; xr[1](1) = (double)(j1+1)/hoplotn;
-				    xr[2](0) = (double)    i1/hoplotn; xr[2](1) = (double)(j1+1)/hoplotn;
-				  };
-				    
-			      for (int l=0; l<3; l++)
-				{
-				  Mat<3,3> dxdxi;
-				  xr[l](2) = (double) trig;
-				  curv.CalcElementTransformation (xr[l], i-1, xg, dxdxi);
-				  for (int i = 0; i < 3; i++)
-				    {
-				      dx(i) = dxdxi(i,0);
-				      dy(i) = dxdxi(i,1);
-				      dz(i) = dxdxi(i,2);
-				    }
+    for (ElementIndex ei = 0; ei < mesh->GetNE(); ei++)
+      {
+	const Element & el = (*mesh)[ei];
+	if (el.GetType() == PRISM && !el.IsDeleted())
+	  {
+	    int j;
+	    int i = ei + 1;
 
-				  Vec<3> d0 = x0(0)*dx + x0(1)*dy + x0(2)*dz;
-				  Vec<3> d1 = x1(0)*dx + x1(1)*dy + x1(2)*dz;
-				  n = Cross (d1, d0);
-				  glNormal3d (n(0), n(1), n(2));
-				  glVertex3d (xg(0), xg(1), xg(2));
-				}
-			    }
+	    CurvedElements & curv = mesh->GetCurvedElements();
+	    if (curv.IsHighOrder() && curv.IsElementCurved(ei))
+	      {
+		const ELEMENT_FACE * faces = MeshTopology :: GetFaces (PRISM);
+		const Point3d * vertices = MeshTopology :: GetVertices (PRISM);
+		
+		Point<3> grid[11][11];
+		Point<3> fpts[4];
+		int order = vispar.subdivisions+1;
+		
+		for (int trig = 0; trig < 2; trig++)
+		  {   
+		    for (int j = 0; j < 3; j++)
+		      fpts[j] = vertices[faces[trig][j]-1];
+		    
+		    static Point<3> c(1.0/3.0, 1.0/3.0, 0.5);
+		    if (vispar.shrink < 1)
+		      for (int j = 0; j < 3; j++)
+			fpts[j] += (1-vispar.shrink) * (c-fpts[j]);
 			
-		    }
+		    for (int ix = 0; ix <= order; ix++)
+		      for (int iy = 0; iy <= order; iy++)
+			{
+			  double lami[3] = 
+			    { (1-double(ix)/order) * (1-double(iy)/order),
+			      (  double(ix)/order) * (1-double(iy)/order),
+			      double(iy)/order };
+			      
+			  Point<3> xl;
+			  for (int l = 0; l < 3; l++)
+			    xl(l) = lami[0] * fpts[0](l) + lami[1] * fpts[1](l) +
+			      lami[2] * fpts[2](l);
+			      
+			  curv.CalcElementTransformation (xl, i-1, grid[ix][iy]);
+			}
+			
+		    for (int j = 0; j <= order; j++)
+		      ToBernstein (order, &grid[j][0], &grid[0][1]-&grid[0][0]);
+		    for (int j = 0; j <= order; j++)
+		      ToBernstein (order, &grid[0][j], &grid[1][0]-&grid[0][0]);
+			
+		    glMap2d(GL_MAP2_VERTEX_3, 
+			    0.0, 1.0, &grid[0][1](0)-&grid[0][0](0), order+1,
+			    0.0, 1.0, &grid[1][0](0)-&grid[0][0](0), order+1,
+			    &grid[0][0](0));
+		    glEnable(GL_MAP2_VERTEX_3);
+		    glEnable(GL_AUTO_NORMAL); 
+			
+		    glMapGrid2f(8, 0.0, 0.999, 8, 0.0, 1.0);
+		    glEvalMesh2(GL_FILL, 0, 8, 0, 8);
+			
+		    glDisable (GL_AUTO_NORMAL);
+		    glDisable (GL_MAP2_VERTEX_3);
+		  }		    
+
+		for (int quad = 2; quad < 5; quad++)
+		  {   
+		    for (int j = 0; j < 4; j++)
+		      fpts[j] = vertices[faces[quad][j]-1];
+		    
+		    static Point<3> c(1.0/3.0, 1.0/3.0, 0.5);
+		    if (vispar.shrink < 1)
+		      for (int j = 0; j < 4; j++)
+			fpts[j] += (1-vispar.shrink) * (c-fpts[j]);
+			
+		    for (int ix = 0; ix <= order; ix++)
+		      for (int iy = 0; iy <= order; iy++)
+			{
+			  double lami[4] = 
+			    { (1-double(ix)/order) * (1-double(iy)/order),
+			      (  double(ix)/order) * (1-double(iy)/order),
+			      (  double(ix)/order) * (  double(iy)/order),
+			      (1-double(ix)/order) * (  double(iy)/order) };
+			      
+			  Point<3> xl;
+			  for (int l = 0; l < 3; l++)
+			    xl(l) = 
+			      lami[0] * fpts[0](l) + lami[1] * fpts[1](l) +
+			      lami[2] * fpts[2](l) + lami[3] * fpts[3](l);
+			      
+			  curv.CalcElementTransformation (xl, ei, grid[ix][iy]);
+			}
+			
+		    for (int j = 0; j <= order; j++)
+		      ToBernstein (order, &grid[j][0], &grid[0][1]-&grid[0][0]);
+		    for (int j = 0; j <= order; j++)
+		      ToBernstein (order, &grid[0][j], &grid[1][0]-&grid[0][0]);
+			
+		    glMap2d(GL_MAP2_VERTEX_3, 
+			    0.0, 1.0, &grid[0][1](0)-&grid[0][0](0), order+1,
+			    0.0, 1.0, &grid[1][0](0)-&grid[0][0](0), order+1,
+			    &grid[0][0](0));
+		    glEnable(GL_MAP2_VERTEX_3);
+		    glEnable(GL_AUTO_NORMAL); 
+			
+		    glMapGrid2f(8, 0.0, 1.0, 8, 0.0, 1.0);
+		    glEvalMesh2(GL_FILL, 0, 8, 0, 8);
+			
+		    glDisable (GL_AUTO_NORMAL);
+		    glDisable (GL_MAP2_VERTEX_3);
+		  }		    
 
-		  glEnd ();
 
-		  glBegin (GL_QUADS);
 
-		  for (int quad = 0; quad<3; quad++)
-		    {   
-		      const Point3d * facepoint = MeshTopology :: GetVertices (PRISM);
 
-		      Vec<3> x0,x1;
-		      int xyz;
 
-		      switch (quad)
-			{
-			case 0:
-			  x0 = facepoint[5] - facepoint[2];
-			  x1 = facepoint[0] - facepoint[2];
-			  xyz = 0;
-			  break;
-			case 1:
-			  x0 = facepoint[4] - facepoint[0];
-			  x1 = facepoint[1] - facepoint[0];
-			  xyz = 0;
-			  break;
-			case 2:
-			  x0 = facepoint[1] - facepoint[2];
-			  x1 = facepoint[5] - facepoint[2];
-			  xyz = 1;
-			  break;
-			}
+		/*
+		int hoplotn = 1 << vispar.subdivisions; 
+		// int hoplotn = curv.GetNVisualSubsecs();
 
-		      x0.Normalize();
-		      x1.Normalize();
+		const Point3d * facepoint = MeshTopology :: GetVertices (TRIG);
+		const ELEMENT_FACE * elface = MeshTopology :: GetFaces(TRIG);
 
-		      Swap (x0,x1);
+		glBegin (GL_TRIANGLES);
 
-		      Point<3> xr[4];
-		      Point<3> xg;
-		      Vec<3> dx, dy, dz, n;
+		for (int trig = 0; trig<2; trig++)
+		  {
+                    
+		    Vec<3> x0,x1,d0,d1;
+		    x0 = facepoint[1] - facepoint[2];
+		    x1 = facepoint[0] - facepoint[2];
+		    x0.Normalize();
+		    x1.Normalize();
+		    if (trig == 1) swap (x0,x1);
+
+		    Point<3> xr[3];
+		    Point<3> xg;
+		    Vec<3> dx, dy, dz, n;
 
-		      for (int i1 = 0; i1 < hoplotn; i1++)
-			for (int j1 = 0; j1 < hoplotn; j1++)
+		    for (int i1 = 0; i1 < hoplotn; i1++)
+		      for (int j1 = 0; j1 < hoplotn-i1; j1++)
+			for (int k = 0; k < 2; k++)
 			  {
-			    xr[0](xyz) = (double)    i1/hoplotn; xr[0](2) = (double)    j1/hoplotn;
-			    xr[1](xyz) = (double)(i1+1)/hoplotn; xr[1](2) = (double)    j1/hoplotn;
-			    xr[2](xyz) = (double)(i1+1)/hoplotn; xr[2](2) = (double)(j1+1)/hoplotn;
-			    xr[3](xyz) = (double)    i1/hoplotn; xr[3](2) = (double)(j1+1)/hoplotn;
+			    if (k == 0)
+			      {
+				xr[0](0) = (double)    i1/hoplotn; xr[0](1) = (double)    j1/hoplotn;
+				xr[1](0) = (double)(i1+1)/hoplotn; xr[1](1) = (double)    j1/hoplotn;
+				xr[2](0) = (double)    i1/hoplotn; xr[2](1) = (double)(j1+1)/hoplotn;
+			      } else
+				{
+				  if (j1 == hoplotn-i1-1) continue;
+				  xr[0](0) = (double)(i1+1)/hoplotn; xr[0](1) = (double)    j1/hoplotn;
+				  xr[1](0) = (double)(i1+1)/hoplotn; xr[1](1) = (double)(j1+1)/hoplotn;
+				  xr[2](0) = (double)    i1/hoplotn; xr[2](1) = (double)(j1+1)/hoplotn;
+				};
 				    
-			    for (int l=0; l<4; l++)
+			    for (int l=0; l<3; l++)
 			      {
-				switch (quad)
-				  {
-				  case 0: xr[l](1) = 0; break;
-				  case 1: xr[l](1) = 1-xr[l](0); break;
-				  case 2: xr[l](0) = 0; break;
-				  }
-
 				Mat<3,3> dxdxi;
+				xr[l](2) = (double) trig;
 				curv.CalcElementTransformation (xr[l], i-1, xg, dxdxi);
 				for (int i = 0; i < 3; i++)
 				  {
@@ -2091,461 +2122,803 @@ void VisualSceneMesh :: BuildPrismList()
 				glVertex3d (xg(0), xg(1), xg(2));
 			      }
 			  }
-		    }
-		  glEnd ();
-		} else
-		  { 
-		    Point3d c(0,0,0);
-		    if (vispar.shrink < 1)
+			
+		  }
+
+		glEnd ();
+
+		glBegin (GL_QUADS);
+
+		for (int quad = 0; quad<3; quad++)
+		  {   
+		    const Point3d * facepoint = MeshTopology :: GetVertices (PRISM);
+
+		    Vec<3> x0,x1;
+		    int xyz;
+
+		    switch (quad)
 		      {
-			for (j = 1; j <= 6; j++)
-			  {
-			    Point3d p = mesh->Point(el.PNum(j));
-			    c.X() += p.X() / 6;
-			    c.Y() += p.Y() / 6;
-			    c.Z() += p.Z() / 6;
-			  }
+		      case 0:
+			x0 = facepoint[5] - facepoint[2];
+			x1 = facepoint[0] - facepoint[2];
+			xyz = 0;
+			break;
+		      case 1:
+			x0 = facepoint[4] - facepoint[0];
+			x1 = facepoint[1] - facepoint[0];
+			xyz = 0;
+			break;
+		      case 2:
+			x0 = facepoint[1] - facepoint[2];
+			x1 = facepoint[5] - facepoint[2];
+			xyz = 1;
+			break;
 		      }
 
-		    glBegin (GL_TRIANGLES);
+		    x0.Normalize();
+		    x1.Normalize();
 
-		    for (j = 1; j <= faces.Size(); j++)
+		    swap (x0,x1);
+
+		    Point<3> xr[4];
+		    Point<3> xg;
+		    Vec<3> dx, dy, dz, n;
+
+		    for (int i1 = 0; i1 < hoplotn; i1++)
+		      for (int j1 = 0; j1 < hoplotn; j1++)
+			{
+			  xr[0](xyz) = (double)    i1/hoplotn; xr[0](2) = (double)    j1/hoplotn;
+			  xr[1](xyz) = (double)(i1+1)/hoplotn; xr[1](2) = (double)    j1/hoplotn;
+			  xr[2](xyz) = (double)(i1+1)/hoplotn; xr[2](2) = (double)(j1+1)/hoplotn;
+			  xr[3](xyz) = (double)    i1/hoplotn; xr[3](2) = (double)(j1+1)/hoplotn;
+				    
+			  for (int l=0; l<4; l++)
+			    {
+			      switch (quad)
+				{
+				case 0: xr[l](1) = 0; break;
+				case 1: xr[l](1) = 1-xr[l](0); break;
+				case 2: xr[l](0) = 0; break;
+				}
+
+			      Mat<3,3> dxdxi;
+			      curv.CalcElementTransformation (xr[l], i-1, xg, dxdxi);
+			      for (int i = 0; i < 3; i++)
+				{
+				  dx(i) = dxdxi(i,0);
+				  dy(i) = dxdxi(i,1);
+				  dz(i) = dxdxi(i,2);
+				}
+
+			      Vec<3> d0 = x0(0)*dx + x0(1)*dy + x0(2)*dz;
+			      Vec<3> d1 = x1(0)*dx + x1(1)*dy + x1(2)*dz;
+			      n = Cross (d1, d0);
+			      glNormal3d (n(0), n(1), n(2));
+			      glVertex3d (xg(0), xg(1), xg(2));
+			    }
+			}
+		  }
+		glEnd ();
+		*/
+	      } 
+	    else
+	      { 
+		Point3d c(0,0,0);
+		if (vispar.shrink < 1)
+		  {
+		    for (j = 1; j <= 6; j++)
 		      {
-			Element2d & face = faces.Elem(j);
-			Point3d lp1 = mesh->Point (el.PNum(face.PNum(1)));
-			Point3d lp2 = mesh->Point (el.PNum(face.PNum(2)));
-			Point3d lp3 = mesh->Point (el.PNum(face.PNum(3)));
-			Vec3d n = Cross (Vec3d (lp1, lp3), Vec3d (lp1, lp2));
-			n /= (n.Length()+1e-12);
-			glNormal3d (n.X(), n.Y(), n.Z());
-			if (vispar.shrink < 1)
-			  {
-			    lp1 = c + vispar.shrink * (lp1 - c);
-			    lp2 = c + vispar.shrink * (lp2 - c);
-			    lp3 = c + vispar.shrink * (lp3 - c);
-			  }
-			glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
-			glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
-			glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
+			Point3d p = mesh->Point(el.PNum(j));
+			c.X() += p.X() / 6;
+			c.Y() += p.Y() / 6;
+			c.Z() += p.Z() / 6;
+		      }
+		  }
+
+		el.GetSurfaceTriangles (faces);
+		glBegin (GL_TRIANGLES);
+		for (j = 1; j <= faces.Size(); j++)
+		  {
+		    Element2d & face = faces.Elem(j);
+		    Point3d lp1 = mesh->Point (el.PNum(face.PNum(1)));
+		    Point3d lp2 = mesh->Point (el.PNum(face.PNum(2)));
+		    Point3d lp3 = mesh->Point (el.PNum(face.PNum(3)));
+		    Vec3d n = Cross (Vec3d (lp1, lp3), Vec3d (lp1, lp2));
+		    n /= (n.Length()+1e-12);
+		    glNormal3d (n.X(), n.Y(), n.Z());
+		    if (vispar.shrink < 1)
+		      {
+			lp1 = c + vispar.shrink * (lp1 - c);
+			lp2 = c + vispar.shrink * (lp2 - c);
+			lp3 = c + vispar.shrink * (lp3 - c);
 		      }
+		    glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
+		    glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
+		    glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
+		  }
 	      
-		    glEnd();
+		glEnd();
+	      }
+	  }
+      }
+    glEndList ();
+  }
+ 
+
+
+
+  void VisualSceneMesh :: BuildHexList()
+  {
+    if (hextimestamp > mesh->GetTimeStamp () &&
+	hextimestamp > vispar.clipplanetimestamp )
+      return;
+
+    if (!lock)
+      {
+	lock = new NgLock (mesh->Mutex());
+	lock -> Lock();
+      }
+
+    hextimestamp = NextTimeStamp();
+
+    if (hexlist) glDeleteLists (hexlist, 1);
+
+    hexlist = glGenLists (1);
+    glNewList (hexlist, GL_COMPILE);
+
+
+    static float hexcol[] = { 1.0f, 1.0f, 0.0f, 1.0f };
+    glLineWidth (1.0f);  
+    glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, hexcol);
+
+    ARRAY<Element2d> faces;
+    int hoplotn = 1 << vispar.subdivisions; 
+
+    for (ElementIndex ei = 0; ei < mesh->GetNE(); ei++)
+      {
+	const Element & el = (*mesh)[ei];
+	if (el.GetType() == HEX && !el.IsDeleted())
+	  {
+	    CurvedElements & curv = mesh->GetCurvedElements();
+	    if (curv.IsHighOrder() && curv.IsElementCurved(ei))
+	      {
+		/* // classical 
+		   glBegin (GL_QUADS);
+		
+		   const ELEMENT_FACE * faces = MeshTopology :: GetFaces (HEX);
+		   const Point3d * vertices = MeshTopology :: GetVertices (HEX);
+
+		   Point<3> grid[33][33];
+		   Vec<3> gridn[33][33];
+		   Point<3> fpts[4];
+		   for (int quad = 0; quad<6; quad++)
+		   {   
+		   for (int j = 0; j < 4; j++)
+		   fpts[j] = vertices[faces[quad][j]-1];
+
+		   static Point<3> c(0.5, 0.5, 0.5);
+		   if (vispar.shrink < 1)
+		   for (int j = 0; j < 4; j++)
+		   fpts[j] += (1-vispar.shrink) * (c-fpts[j]);
+
+		   Vec<3> taux = fpts[1]-fpts[0];
+		   Vec<3> tauy = fpts[3]-fpts[0];
+
+		   for (int ix = 0; ix <= hoplotn; ix++)
+		   for (int iy = 0; iy <= hoplotn; iy++)
+		   {
+		   Point<3> xl;
+		   Mat<3,3> dxdxi;
+		   double lami[4] = 
+		   { (1-double(ix)/hoplotn) * (1-double(iy)/hoplotn),
+		   (  double(ix)/hoplotn) * (1-double(iy)/hoplotn),
+		   (  double(ix)/hoplotn) * (  double(iy)/hoplotn),
+		   (1-double(ix)/hoplotn) * (  double(iy)/hoplotn) };
+		   for (int l = 0; l < 3; l++)
+		   xl(l) = lami[0] * fpts[0](l) + lami[1] * fpts[1](l) +
+		   lami[2] * fpts[2](l) + lami[3] * fpts[3](l);
+
+		   curv.CalcElementTransformation (xl, ei, grid[ix][iy], dxdxi);
+			  
+		   Vec<3> gtaux = dxdxi * taux;
+		   Vec<3> gtauy = dxdxi * tauy;
+		   gridn[ix][iy] = Cross (gtauy, gtaux).Normalize();
+		   }
+		    
+		   for (int ix = 0; ix < hoplotn; ix++)
+		   for (int iy = 0; iy < hoplotn; iy++)
+		   {
+		   glNormal3dv (gridn[ix][iy]);
+		   glVertex3dv (grid[ix][iy]);
+
+		   glNormal3dv (gridn[ix+1][iy]);
+		   glVertex3dv (grid[ix+1][iy]);
+
+		   glNormal3dv (gridn[ix+1][iy+1]);
+		   glVertex3dv (grid[ix+1][iy+1]);
+
+		   glNormal3dv (gridn[ix][iy+1]);
+		   glVertex3dv (grid[ix][iy+1]);
+		   }
+		   }
+		
+		   glEnd ();
+		*/
+
+		const ELEMENT_FACE * faces = MeshTopology :: GetFaces (HEX);
+		const Point3d * vertices = MeshTopology :: GetVertices (HEX);
+
+		Point<3> grid[11][11];
+		Point<3> fpts[4];
+		int order = vispar.subdivisions+1;
+
+		for (int quad = 0; quad<6; quad++)
+		  {   
+		    for (int j = 0; j < 4; j++)
+		      fpts[j] = vertices[faces[quad][j]-1];
+
+		    static Point<3> c(0.5, 0.5, 0.5);
+		    if (vispar.shrink < 1)
+		      for (int j = 0; j < 4; j++)
+			fpts[j] += (1-vispar.shrink) * (c-fpts[j]);
+
+		    for (int ix = 0; ix <= order; ix++)
+		      for (int iy = 0; iy <= order; iy++)
+			{
+			  double lami[4] = 
+			    { (1-double(ix)/order) * (1-double(iy)/order),
+			      (  double(ix)/order) * (1-double(iy)/order),
+			      (  double(ix)/order) * (  double(iy)/order),
+			      (1-double(ix)/order) * (  double(iy)/order) };
+
+			  Point<3> xl;
+			  for (int l = 0; l < 3; l++)
+			    xl(l) = lami[0] * fpts[0](l) + lami[1] * fpts[1](l) +
+			      lami[2] * fpts[2](l) + lami[3] * fpts[3](l);
+
+			  curv.CalcElementTransformation (xl, ei, grid[ix][iy]);
+			}
+		    
+		    for (int j = 0; j <= order; j++)
+		      ToBernstein (order, &grid[j][0], &grid[0][1]-&grid[0][0]);
+		    for (int j = 0; j <= order; j++)
+		      ToBernstein (order, &grid[0][j], &grid[1][0]-&grid[0][0]);
+
+		    glMap2d(GL_MAP2_VERTEX_3, 
+			    0.0, 1.0, &grid[0][1](0)-&grid[0][0](0), order+1,
+			    0.0, 1.0, &grid[1][0](0)-&grid[0][0](0), order+1,
+			    &grid[0][0](0));
+		    glEnable(GL_MAP2_VERTEX_3);
+		    glEnable(GL_AUTO_NORMAL); 
+
+		    glMapGrid2f(8, 0.0, 1.0, 8, 0.0, 1.0);
+		    glEvalMesh2(GL_FILL, 0, 8, 0, 8);
+
+		    glDisable (GL_AUTO_NORMAL);
+		    glDisable (GL_MAP2_VERTEX_3);
 		  }
-	    }
-	}
-    }
-  glEndList ();
-}
+	      } 
+	    else
+	      { 
+		Point3d c(0,0,0);
+		if (vispar.shrink < 1)
+		  {
+		    for (int j = 1; j <= 8; j++)
+		      {
+			Point3d p = mesh->Point(el.PNum(j));
+			c.X() += p.X();
+			c.Y() += p.Y();
+			c.Z() += p.Z();
+		      }
+		    c.X() /= 8;
+		    c.Y() /= 8;
+		    c.Z() /= 8;
+		  }
+
+		glBegin (GL_TRIANGLES);
+
+		el.GetSurfaceTriangles (faces);
+		for (int j = 1; j <= faces.Size(); j++)
+		  {
+		    Element2d & face = faces.Elem(j);
+		    Point<3> lp1 = mesh->Point (el.PNum(face.PNum(1)));
+		    Point<3> lp2 = mesh->Point (el.PNum(face.PNum(2)));
+		    Point<3> lp3 = mesh->Point (el.PNum(face.PNum(3)));
+		    Vec<3> n = Cross (lp3-lp1, lp2-lp1);  
+		    n.Normalize();
+		    glNormal3dv (n);
+		  
+		    if (vispar.shrink < 1)
+		      {
+			lp1 = c + vispar.shrink * (lp1 - c);
+			lp2 = c + vispar.shrink * (lp2 - c);
+			lp3 = c + vispar.shrink * (lp3 - c);
+		      }
+
+		    glVertex3dv (lp1);
+		    glVertex3dv (lp2);
+		    glVertex3dv (lp3);
+		  }
+
+		glEnd();
+	      }
+	  }
+      }
+    glEndList ();
+  }
+
+
+
+
+
+
+
+
  
+  void VisualSceneMesh :: BuildPyramidList()
+  {
+    if (pyramidtimestamp > mesh->GetTimeStamp () &&
+	pyramidtimestamp > vispar.clipplanetimestamp )
+      return;
+
+    if (!lock)
+      {
+	lock = new NgLock (mesh->Mutex());
+	lock -> Lock();
+      }
 
+    pyramidtimestamp = NextTimeStamp();
 
 
+    if (pyramidlist)
+      glDeleteLists (pyramidlist, 1);
 
 
 
-void VisualSceneMesh :: BuildHexList()
-{
-  int i, j;
 
-  hexlist = glGenLists (1);
-  glNewList (hexlist, GL_COMPILE);
+    pyramidlist = glGenLists (1);
+    glNewList (pyramidlist, GL_COMPILE);
+  
+    static float pyramidcol[] = { 1.0f, 0.0f, 1.0f, 1.0f };
+    glLineWidth (1.0f);
+    ARRAY<Element2d> faces;
+
+    for (ElementIndex ei = 0; ei < mesh->GetNE(); ei++)
+      {
+	const Element & el = (*mesh)[ei];
+	if (el.GetType() == PYRAMID && !el.IsDeleted())
+	  {
+	    int j;
+	    int i = ei + 1;
+
+	    CurvedElements & curv = mesh->GetCurvedElements();
+	    if (curv.IsHighOrder() && curv.IsElementCurved(ei))
+	      {
 
-  static float hexcol[] = { 1.0f, 1.0f, 0.0f, 1.0f };
-  glLineWidth (1.0f);  
-  glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, hexcol);
+		const ELEMENT_FACE * faces = MeshTopology :: GetFaces (PYRAMID);
+		const Point3d * vertices = MeshTopology :: GetVertices (PYRAMID);
+		
+		Point<3> grid[11][11];
+		Point<3> fpts[4];
+		int order = vispar.subdivisions+1;
+		
+		for (int trig = 0; trig < 4; trig++)
+		  {   
+		    for (int j = 0; j < 3; j++)
+		      fpts[j] = vertices[faces[trig][j]-1];
+		    
+		    static Point<3> c(0.375, 0.375, 0.25);
+		    if (vispar.shrink < 1)
+		      for (int j = 0; j < 3; j++)
+			fpts[j] += (1-vispar.shrink) * (c-fpts[j]);
+			
+		    for (int ix = 0; ix <= order; ix++)
+		      for (int iy = 0; iy <= order; iy++)
+			{
+			  double lami[3] = 
+			    { (1-double(ix)/order) * (1-double(iy)/order),
+			      (  double(ix)/order) * (1-double(iy)/order),
+			      double(iy)/order };
+			      
+			  Point<3> xl;
+			  for (int l = 0; l < 3; l++)
+			    xl(l) = lami[0] * fpts[0](l) + lami[1] * fpts[1](l) +
+			      lami[2] * fpts[2](l);
+			      
+			  curv.CalcElementTransformation (xl, i-1, grid[ix][iy]);
+			}
+			
+		    for (int j = 0; j <= order; j++)
+		      ToBernstein (order, &grid[j][0], &grid[0][1]-&grid[0][0]);
+		    for (int j = 0; j <= order; j++)
+		      ToBernstein (order, &grid[0][j], &grid[1][0]-&grid[0][0]);
+			
+		    glMap2d(GL_MAP2_VERTEX_3, 
+			    0.0, 1.0, &grid[0][1](0)-&grid[0][0](0), order+1,
+			    0.0, 1.0, &grid[1][0](0)-&grid[0][0](0), order+1,
+			    &grid[0][0](0));
+		    glEnable(GL_MAP2_VERTEX_3);
+		    glEnable(GL_AUTO_NORMAL); 
+			
+		    glMapGrid2f(8, 0.0, 0.999, 8, 0.0, 1.0);
+		    glEvalMesh2(GL_FILL, 0, 8, 0, 8);
+			
+		    glDisable (GL_AUTO_NORMAL);
+		    glDisable (GL_MAP2_VERTEX_3);
+		  }		    
+
+		for (int quad = 4; quad < 5; quad++)
+		  {   
+		    for (int j = 0; j < 4; j++)
+		      fpts[j] = vertices[faces[quad][j]-1];
+		    
+		    static Point<3> c(0.375, 0.375, 0.25);
+		    if (vispar.shrink < 1)
+		      for (int j = 0; j < 4; j++)
+			fpts[j] += (1-vispar.shrink) * (c-fpts[j]);
+			
+		    for (int ix = 0; ix <= order; ix++)
+		      for (int iy = 0; iy <= order; iy++)
+			{
+			  double lami[4] = 
+			    { (1-double(ix)/order) * (1-double(iy)/order),
+			      (  double(ix)/order) * (1-double(iy)/order),
+			      (  double(ix)/order) * (  double(iy)/order),
+			      (1-double(ix)/order) * (  double(iy)/order) };
+			      
+			  Point<3> xl;
+			  for (int l = 0; l < 3; l++)
+			    xl(l) = 
+			      lami[0] * fpts[0](l) + lami[1] * fpts[1](l) +
+			      lami[2] * fpts[2](l) + lami[3] * fpts[3](l);
+			      
+			  curv.CalcElementTransformation (xl, ei, grid[ix][iy]);
+			}
+			
+		    for (int j = 0; j <= order; j++)
+		      ToBernstein (order, &grid[j][0], &grid[0][1]-&grid[0][0]);
+		    for (int j = 0; j <= order; j++)
+		      ToBernstein (order, &grid[0][j], &grid[1][0]-&grid[0][0]);
+			
+		    glMap2d(GL_MAP2_VERTEX_3, 
+			    0.0, 1.0, &grid[0][1](0)-&grid[0][0](0), order+1,
+			    0.0, 1.0, &grid[1][0](0)-&grid[0][0](0), order+1,
+			    &grid[0][0](0));
+		    glEnable(GL_MAP2_VERTEX_3);
+		    glEnable(GL_AUTO_NORMAL); 
+			
+		    glMapGrid2f(8, 0.0, 1.0, 8, 0.0, 1.0);
+		    glEvalMesh2(GL_FILL, 0, 8, 0, 8);
+			
+		    glDisable (GL_AUTO_NORMAL);
+		    glDisable (GL_MAP2_VERTEX_3);
+		  }		    
 
-  ARRAY<Element2d> faces;
 
-  for (i = 1; i <= mesh->GetNE(); i++)
-    {
-      Element el = mesh->VolumeElement (i);
 
-      if (el.GetType() == HEX &&
-	  el.PNum(1))
-	{
-	  el.GetSurfaceTriangles (faces);
 
-	  CurvedElements & curv = mesh->GetCurvedElements();
-	  if (curv.IsHighOrder() && curv.IsElementCurved(i-1))
-	    {
-	      cerr << "draw curved hex not implemented" << endl;
-	      int hoplotn = 1 << vispar.subdivisions; 
-	      // int hoplotn = curv.GetNVisualSubsecs();
-	      
-	      const Point3d * facepoint = MeshTopology :: GetVertices (TRIG);
-	      const ELEMENT_FACE * elface = MeshTopology :: GetFaces(TRIG);
-	      
-	      glBegin (GL_TRIANGLES);
-	      
-	      for (int trig = 0; trig<2; trig++)
-		{
-		  Vec<3> x0,x1,d0,d1;
-		  x0 = facepoint[1] - facepoint[2];
-		  x1 = facepoint[0] - facepoint[2];
-		      x0.Normalize();
-		      x1.Normalize();
-		      if (trig == 1) Swap (x0,x1);
 
-		      Point<3> xr[3];
-		      Point<3> xg;
-		      Vec<3> dx, dy, dz, n;
 
-		      for (int i1 = 0; i1 < hoplotn; i1++)
-			for (int j1 = 0; j1 < hoplotn-i1; j1++)
-			  for (int k = 0; k < 2; k++)
-			    {
-			      if (k == 0)
-				{
-				  xr[0](0) = (double)    i1/hoplotn; xr[0](1) = (double)    j1/hoplotn;
-				  xr[1](0) = (double)(i1+1)/hoplotn; xr[1](1) = (double)    j1/hoplotn;
-				  xr[2](0) = (double)    i1/hoplotn; xr[2](1) = (double)(j1+1)/hoplotn;
-				} else
-				  {
-				    if (j1 == hoplotn-i1-1) continue;
-				    xr[0](0) = (double)(i1+1)/hoplotn; xr[0](1) = (double)    j1/hoplotn;
-				    xr[1](0) = (double)(i1+1)/hoplotn; xr[1](1) = (double)(j1+1)/hoplotn;
-				    xr[2](0) = (double)    i1/hoplotn; xr[2](1) = (double)(j1+1)/hoplotn;
-				  };
-				    
-			      for (int l=0; l<3; l++)
-				{
-				  Mat<3,3> dxdxi;
-				  xr[l](2) = (double) trig;
-				  curv.CalcElementTransformation (xr[l], i-1, xg, dxdxi);
-				  for (int i = 0; i < 3; i++)
-				    {
-				      dx(i) = dxdxi(i,0);
-				      dy(i) = dxdxi(i,1);
-				      dz(i) = dxdxi(i,2);
-				    }
+		/*
+		int hoplotn = 1 << vispar.subdivisions; 
 
-				  Vec<3> d0 = x0(0)*dx + x0(1)*dy + x0(2)*dz;
-				  Vec<3> d1 = x1(0)*dx + x1(1)*dy + x1(2)*dz;
-				  n = Cross (d1, d0);
-				  glNormal3d (n(0), n(1), n(2));
-				  glVertex3d (xg(0), xg(1), xg(2));
-				}
-			    }
-			
-		    }
+		const ELEMENT_FACE * faces = MeshTopology :: GetFaces (PYRAMID);
+		const Point3d * vertices = MeshTopology :: GetVertices (PYRAMID);
 
-		  glEnd ();
+		Point<3> grid[33][33];
+		Vec<3> gridn[33][33];
 
-		  glBegin (GL_QUADS);
 
-		  for (int quad = 0; quad<3; quad++)
-		    {   
-		      const Point3d * facepoint = MeshTopology :: GetVertices (PRISM);
+		glBegin (GL_TRIANGLES);
+		
+		for (int trig = 0; trig < 4; trig++)
+		  {   
+		    Point<3> p0 = vertices[faces[trig][0]-1];
+		    Point<3> p1 = vertices[faces[trig][1]-1];
+		    Point<3> p2 = vertices[faces[trig][2]-1];
+
+		    if (vispar.shrink < 1)
+		      {
+			static Point<3> c(0.375, 0.375, 0.25);
+			p0 = c + vispar.shrink * (p0 - c);
+			p1 = c + vispar.shrink * (p1 - c);
+			p2 = c + vispar.shrink * (p2 - c);
+		      }
+		   
 
-		      Vec<3> x0,x1;
-		      int xyz;
+		    Vec<3> taux = p0-p2;
+		    Vec<3> tauy = p1-p2;
+		    Vec<3> gtaux, gtauy;
 
-		      switch (quad)
+		    Point<3> xl;
+		    Mat<3,3> dxdxi;
+
+		    for (int ix = 0; ix <= hoplotn; ix++)
+		      for (int iy = 0; iy <= hoplotn-ix; iy++)
 			{
-			case 0:
-			  x0 = facepoint[5] - facepoint[2];
-			  x1 = facepoint[0] - facepoint[2];
-			  xyz = 0;
-			  break;
-			case 1:
-			  x0 = facepoint[4] - facepoint[0];
-			  x1 = facepoint[1] - facepoint[0];
-			  xyz = 0;
-			  break;
-			case 2:
-			  x0 = facepoint[1] - facepoint[2];
-			  x1 = facepoint[5] - facepoint[2];
-			  xyz = 1;
-			  break;
+			  for (int l = 0; l < 3; l++)
+			    xl(l) = 
+			      (1-double(ix+iy)/hoplotn) * p2(l) +
+			      (double(ix)/hoplotn) * p0(l) +
+			      (double(iy)/hoplotn) * p1(l);
+			  
+			  curv.CalcElementTransformation (xl, i-1, grid[ix][iy], dxdxi);
+			  
+			  gtaux = dxdxi * taux;
+			  gtauy = dxdxi * tauy;
+			  gridn[ix][iy] = Cross (gtauy, gtaux).Normalize();
 			}
 
-		      x0.Normalize();
-		      x1.Normalize();
+		    for (int ix = 0; ix < hoplotn; ix++)
+		      for (int iy = 0; iy < hoplotn-ix; iy++)
+			{
+			  glNormal3dv (gridn[ix][iy]);
+			  glVertex3dv (grid[ix][iy]);
 
-		      Swap (x0,x1);
+			  glNormal3dv (gridn[ix+1][iy]);
+			  glVertex3dv (grid[ix+1][iy]);
 
-		      Point<3> xr[4];
-		      Point<3> xg;
-		      Vec<3> dx, dy, dz, n;
+			  glNormal3dv (gridn[ix][iy+1]);
+			  glVertex3dv (grid[ix][iy+1]);
 
-		      for (int i1 = 0; i1 < hoplotn; i1++)
-			for (int j1 = 0; j1 < hoplotn; j1++)
-			  {
-			    xr[0](xyz) = (double)    i1/hoplotn; xr[0](2) = (double)    j1/hoplotn;
-			    xr[1](xyz) = (double)(i1+1)/hoplotn; xr[1](2) = (double)    j1/hoplotn;
-			    xr[2](xyz) = (double)(i1+1)/hoplotn; xr[2](2) = (double)(j1+1)/hoplotn;
-			    xr[3](xyz) = (double)    i1/hoplotn; xr[3](2) = (double)(j1+1)/hoplotn;
-				    
-			    for (int l=0; l<4; l++)
-			      {
-				switch (quad)
-				  {
-				  case 0: xr[l](1) = 0; break;
-				  case 1: xr[l](1) = 1-xr[l](0); break;
-				  case 2: xr[l](0) = 0; break;
-				  }
+			  if (iy < hoplotn-ix-1)
+			    {
+			      glNormal3dv (gridn[ix][iy+1]);
+			      glVertex3dv (grid[ix][iy+1]);
+			      
+			      glNormal3dv (gridn[ix+1][iy]);
+			      glVertex3dv (grid[ix+1][iy]);
+
+			      glNormal3dv (gridn[ix+1][iy+1]);
+			      glVertex3dv (grid[ix+1][iy+1]);
+			    }
+			}
+		  }
 
-				Mat<3,3> dxdxi;
-				curv.CalcElementTransformation (xr[l], i-1, xg, dxdxi);
-				for (int i = 0; i < 3; i++)
-				  {
-				    dx(i) = dxdxi(i,0);
-				    dy(i) = dxdxi(i,1);
-				    dz(i) = dxdxi(i,2);
-				  }
+		glEnd ();
 
-				Vec<3> d0 = x0(0)*dx + x0(1)*dy + x0(2)*dz;
-				Vec<3> d1 = x1(0)*dx + x1(1)*dy + x1(2)*dz;
-				n = Cross (d1, d0);
-				glNormal3d (n(0), n(1), n(2));
-				glVertex3d (xg(0), xg(1), xg(2));
-			      }
-			  }
-		    }
-		  glEnd ();
-	    } 
-	  else
-	    { 
-	      Point3d c(0,0,0);
-	      if (vispar.shrink < 1)
-		{
-		  for (j = 1; j <= 8; j++)
-		    {
-		      Point3d p = mesh->Point(el.PNum(j));
-		      c.X() += p.X();
-		      c.Y() += p.Y();
-		      c.Z() += p.Z();
-		    }
-		  c.X() /= 8;
-		  c.Y() /= 8;
-		  c.Z() /= 8;
-		}
 
 
-	      glBegin (GL_TRIANGLES);
-	      
-	      for (j = 1; j <= faces.Size(); j++)
-		{
-		  Element2d & face = faces.Elem(j);
-		  Point3d lp1 = mesh->Point (el.PNum(face.PNum(1)));
-		  Point3d lp2 = mesh->Point (el.PNum(face.PNum(2)));
-		  Point3d lp3 = mesh->Point (el.PNum(face.PNum(3)));
-		  Vec3d n = Cross (Vec3d (lp1, lp3), Vec3d (lp1, lp2));
-		  n /= (n.Length()+1e-12);
-		  glNormal3d (n.X(), n.Y(), n.Z());
-		  
-		  if (vispar.shrink < 1)
-		    {
-		      lp1 = c + vispar.shrink * (lp1 - c);
-		      lp2 = c + vispar.shrink * (lp2 - c);
-		      lp3 = c + vispar.shrink * (lp3 - c);
-		    }
 
-		  glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
-		  glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
-		  glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
-		}
-	      glEnd();
-	    }
-	}
-    }
-  glEndList ();
-}
+		glBegin (GL_QUADS);
+		
+		for (int quad = 4; quad < 5; quad++)
+		  {   
+		    Point<3> p0 = vertices[faces[quad][0]-1];
+		    Point<3> p1 = vertices[faces[quad][1]-1];
+		    Point<3> p2 = vertices[faces[quad][2]-1];
+		    Point<3> p3 = vertices[faces[quad][3]-1];
 
+		    if (vispar.shrink < 1)
+		      {
+			static Point<3> c(0.375, 0.375, 0.25);
+			p0 = c + vispar.shrink * (p0 - c);
+			p1 = c + vispar.shrink * (p1 - c);
+			p2 = c + vispar.shrink * (p2 - c);
+			p3 = c + vispar.shrink * (p3 - c);
+		      }
 
+		    Vec<3> taux = p1-p0;
+		    Vec<3> tauy = p3-p0;
+		    Vec<3> gtaux, gtauy;
 
+		    Point<3> xl, xg;
+		    Mat<3,3> dxdxi;
 
+		    for (int ix = 0; ix <= hoplotn; ix++)
+		      for (int iy = 0; iy <= hoplotn; iy++)
+			{
+			  Point<3> xl;
+			  for (int l = 0; l < 3; l++)
+			    xl(l) = 
+			      (1-double(ix)/hoplotn)*(1-double(iy)/hoplotn) * p0(l) +
+			      (  double(ix)/hoplotn)*(1-double(iy)/hoplotn) * p1(l) +
+			      (  double(ix)/hoplotn)*(  double(iy)/hoplotn) * p2(l) +
+			      (1-double(ix)/hoplotn)*(  double(iy)/hoplotn) * p3(l);
+			  
+			  curv.CalcElementTransformation (xl, i-1, grid[ix][iy], dxdxi);
+			  
+			  gtaux = dxdxi * taux;
+			  gtauy = dxdxi * tauy;
+			  gridn[ix][iy] = Cross (gtauy, gtaux).Normalize();
+			}
 
+		    for (int ix = 0; ix < hoplotn; ix++)
+		      for (int iy = 0; iy < hoplotn; iy++)
+			{
+			  glNormal3dv (gridn[ix][iy]);
+			  glVertex3dv (grid[ix][iy]);
 
+			  glNormal3dv (gridn[ix+1][iy]);
+			  glVertex3dv (grid[ix+1][iy]);
 
+			  glNormal3dv (gridn[ix+1][iy+1]);
+			  glVertex3dv (grid[ix+1][iy+1]);
 
- 
-void VisualSceneMesh :: BuildPyramidList()
-{
-  int i, j;
+			  glNormal3dv (gridn[ix][iy+1]);
+			  glVertex3dv (grid[ix][iy+1]);
+			}
+		  }
+
+		glEnd ();
+		*/
+
+
+	      } 
+	    else
+	      { 
 
-  pyramidlist = glGenLists (1);
-  glNewList (pyramidlist, GL_COMPILE);
-  
-  static float pyramidcol[] = { 1.0f, 0.0f, 1.0f, 1.0f };
-  glLineWidth (1.0f);
-  ARRAY<Element2d> faces;
-
-  for (i = 1; i <= mesh->GetNE(); i++)
-    {
-      if (mesh->VolumeElement(i).GetType() == PYRAMID)
-	{
-	  // copy to be thread-safe
-	  Element el = mesh->VolumeElement (i);
-
-	  Point3d c(0,0,0);
-	  if (vispar.shrink < 1)
-	    {
-	      for (j = 1; j <= 5; j++)
-		{
-		  Point3d p = mesh->Point(el.PNum(j));
-		  c.X() += p.X() / 5;
-		  c.Y() += p.Y() / 5;
-		  c.Z() += p.Z() / 5;
-		}
-	    }
 
 
-	  el.GetSurfaceTriangles (faces);
+		Point3d c(0,0,0);
+		if (vispar.shrink < 1)
+		  {
+		    for (int j = 1; j <= 5; j++)
+		      {
+			Point3d p = mesh->Point(el.PNum(j));
+			c.X() += p.X() / 5;
+			c.Y() += p.Y() / 5;
+			c.Z() += p.Z() / 5;
+		      }
+		  }
+
+
+		el.GetSurfaceTriangles (faces);
 	  
-	  glMaterialfv(GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, pyramidcol);
-	  if (el.PNum(1))
-	    {
-	      glBegin (GL_TRIANGLES);
+		glMaterialfv(GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, pyramidcol);
+		if (el.PNum(1))
+		  {
+		    glBegin (GL_TRIANGLES);
 	      
-	      for (j = 1; j <= faces.Size(); j++)
-		{
-		  Element2d & face = faces.Elem(j);
-		  Point3d lp1 = mesh->Point (el.PNum(face.PNum(1)));
-		  Point3d lp2 = mesh->Point (el.PNum(face.PNum(2)));
-		  Point3d lp3 = mesh->Point (el.PNum(face.PNum(3)));
-		  Vec3d n = Cross (Vec3d (lp1, lp2), Vec3d (lp1, lp3));
-		  n /= (n.Length()+1e-12);
-		  n *= -1;
-		  glNormal3d (n.X(), n.Y(), n.Z());
+		    for (int j = 1; j <= faces.Size(); j++)
+		      {
+			Element2d & face = faces.Elem(j);
+			Point3d lp1 = mesh->Point (el.PNum(face.PNum(1)));
+			Point3d lp2 = mesh->Point (el.PNum(face.PNum(2)));
+			Point3d lp3 = mesh->Point (el.PNum(face.PNum(3)));
+			Vec3d n = Cross (Vec3d (lp1, lp2), Vec3d (lp1, lp3));
+			n /= (n.Length()+1e-12);
+			n *= -1;
+			glNormal3d (n.X(), n.Y(), n.Z());
 
-		  if (vispar.shrink < 1)
-		    {
-		      lp1 = c + vispar.shrink * (lp1 - c);
-		      lp2 = c + vispar.shrink * (lp2 - c);
-		      lp3 = c + vispar.shrink * (lp3 - c);
-		    }
+			if (vispar.shrink < 1)
+			  {
+			    lp1 = c + vispar.shrink * (lp1 - c);
+			    lp2 = c + vispar.shrink * (lp2 - c);
+			    lp3 = c + vispar.shrink * (lp3 - c);
+			  }
 
-		  glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
-		  glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
-		  glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
-		}
+			glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
+			glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
+			glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
+		      }
 	      
-	      glEnd();
-	    }
-	}
-    }
-  glEndList ();
-}
+		    glEnd();
+		  }
+	      }
+	  }
+      }
+    glEndList ();
+  }
 
-void VisualSceneMesh :: BuildBadelList()
-{
-  ;
-}
+  void VisualSceneMesh :: BuildBadelList()
+  {
+    ;
+  }
 
-void VisualSceneMesh :: BuildIdentifiedList()
-{
-  ;
-}
+  void VisualSceneMesh :: BuildIdentifiedList()
+  {
+    ;
+  }
   
-void VisualSceneMesh :: BuildDomainSurfList()
-{
-  if (domainsurflist)
-    glDeleteLists (domainsurflist, 1);
+  void VisualSceneMesh :: BuildDomainSurfList()
+  {
+    if (domainsurflist)
+      glDeleteLists (domainsurflist, 1);
 
-  domainsurflist = glGenLists (1);
-  glNewList (domainsurflist, GL_COMPILE);
+    domainsurflist = glGenLists (1);
+    glNewList (domainsurflist, GL_COMPILE);
 
-  int i, j;
-  glLineWidth (1.0f);
+    int i, j;
+    glLineWidth (1.0f);
   
-  glDisable (GL_COLOR_MATERIAL);
+    glDisable (GL_COLOR_MATERIAL);
   
-  for (i = 1; i <= mesh->GetNSE(); i++)
-    {
-      Element2d el = mesh->SurfaceElement (i);
+    for (i = 1; i <= mesh->GetNSE(); i++)
+      {
+	Element2d el = mesh->SurfaceElement (i);
       
-      int drawel = 1;
-      for (j = 1; j <= el.GetNP(); j++)
-	{
-	  if (!el.PNum(j))
-	    drawel = 0;
-	}
+	int drawel = 1;
+	for (j = 1; j <= el.GetNP(); j++)
+	  {
+	    if (!el.PNum(j))
+	      drawel = 0;
+	  }
       
-      if (!drawel)
-	continue;
+	if (!drawel)
+	  continue;
       
-      if (el.GetIndex() < 1 || el.GetIndex() > mesh->GetNFD())
-	continue;
-      int domin = mesh->GetFaceDescriptor(el.GetIndex()).DomainIn();
-      int domout = mesh->GetFaceDescriptor(el.GetIndex()).DomainOut();
-
-      int fac;
-      if (domin == vispar.drawdomainsurf)
-	fac = 1;
-      else if (domout == vispar.drawdomainsurf)
-	fac = -1;
-      else
-	continue;
+	if (el.GetIndex() < 1 || el.GetIndex() > mesh->GetNFD())
+	  continue;
+	int domin = mesh->GetFaceDescriptor(el.GetIndex()).DomainIn();
+	int domout = mesh->GetFaceDescriptor(el.GetIndex()).DomainOut();
+
+	int fac;
+	if (domin == vispar.drawdomainsurf)
+	  fac = 1;
+	else if (domout == vispar.drawdomainsurf)
+	  fac = -1;
+	else
+	  continue;
       
       
-      GLfloat matcol[] = { 1, 0, 0, 1 };
-      glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, matcol);
+	GLfloat matcol[] = { 1, 0, 0, 1 };
+	glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, matcol);
       
       
-      if (el.GetNP() == 3)
-	{
-	  glBegin (GL_TRIANGLES);
+	if (el.GetNP() == 3)
+	  {
+	    glBegin (GL_TRIANGLES);
 	      
-	  const Point3d & lp1 = mesh->Point (el.PNum(1));
-	  const Point3d & lp2 = mesh->Point (el.PNum(2));
-	  const Point3d & lp3 = mesh->Point (el.PNum(3));
-	  Vec3d n = Cross (Vec3d (lp1, lp2), Vec3d (lp1, lp3));
-	  n /= ( fac * (n.Length()+1e-12));
-	  glNormal3d (n.X(), n.Y(), n.Z());
+	    const Point3d & lp1 = mesh->Point (el.PNum(1));
+	    const Point3d & lp2 = mesh->Point (el.PNum(2));
+	    const Point3d & lp3 = mesh->Point (el.PNum(3));
+	    Vec3d n = Cross (Vec3d (lp1, lp2), Vec3d (lp1, lp3));
+	    n /= ( fac * (n.Length()+1e-12));
+	    glNormal3d (n.X(), n.Y(), n.Z());
 	  
-	  if (!vispar.colormeshsize)
-	    {
-	      glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
-	      glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
-	      glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
-	    }
-	  glEnd();
-	}
-      else if (el.GetNP() == 4)
-	{
-	  glBegin (GL_QUADS);
+	    if (!vispar.colormeshsize)
+	      {
+		glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
+		glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
+		glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
+	      }
+	    glEnd();
+	  }
+	else if (el.GetNP() == 4)
+	  {
+	    glBegin (GL_QUADS);
 	  
-	  const Point3d & lp1 = mesh->Point (el.PNum(1));
-	  const Point3d & lp2 = mesh->Point (el.PNum(2));
-	  const Point3d & lp3 = mesh->Point (el.PNum(4));
-	  const Point3d & lp4 = mesh->Point (el.PNum(3));
-	  Vec3d n = Cross (Vec3d (lp1, lp2), 
-			   Vec3d (lp1, Center (lp3, lp4)));
-	  n /= (fac * (n.Length()+1e-12));
-	  glNormal3d (n.X(), n.Y(), n.Z()); 
-	  glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
-	  glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
-	  glVertex3d (lp4.X(), lp4.Y(), lp4.Z());
-	  glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
-	  glEnd();
-	}
-      else if (el.GetNP() == 6)
-	{
-	  glBegin (GL_TRIANGLES);
-	  static int trigs[4][3] = {
-	    { 1, 6, 5 },
-	    { 2, 4, 6 },
-	    { 3, 5, 4 },
-	    { 4, 5, 6 } };
+	    const Point3d & lp1 = mesh->Point (el.PNum(1));
+	    const Point3d & lp2 = mesh->Point (el.PNum(2));
+	    const Point3d & lp3 = mesh->Point (el.PNum(4));
+	    const Point3d & lp4 = mesh->Point (el.PNum(3));
+	    Vec3d n = Cross (Vec3d (lp1, lp2), 
+			     Vec3d (lp1, Center (lp3, lp4)));
+	    n /= (fac * (n.Length()+1e-12));
+	    glNormal3d (n.X(), n.Y(), n.Z()); 
+	    glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
+	    glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
+	    glVertex3d (lp4.X(), lp4.Y(), lp4.Z());
+	    glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
+	    glEnd();
+	  }
+	else if (el.GetNP() == 6)
+	  {
+	    glBegin (GL_TRIANGLES);
+	    static int trigs[4][3] = {
+	      { 1, 6, 5 },
+	      { 2, 4, 6 },
+	      { 3, 5, 4 },
+	      { 4, 5, 6 } };
 	  
-	  for (j = 0; j < 4; j++)
-	    {
-	      const Point3d & lp1 = mesh->Point (el.PNum(trigs[j][0]));
-	      const Point3d & lp2 = mesh->Point (el.PNum(trigs[j][1]));
-	      const Point3d & lp3 = mesh->Point (el.PNum(trigs[j][2]));
-	      Vec3d n = Cross (Vec3d (lp1, lp2), Vec3d (lp1, lp3));
-	      n /= (fac * (n.Length() + 1e-12));
-	      glNormal3d (n.X(), n.Y(), n.Z());
-	      glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
-	      glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
-	      glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
-	    }
-	  glEnd();
-	}
-    }
-  glEndList ();
-}
+	    for (j = 0; j < 4; j++)
+	      {
+		const Point3d & lp1 = mesh->Point (el.PNum(trigs[j][0]));
+		const Point3d & lp2 = mesh->Point (el.PNum(trigs[j][1]));
+		const Point3d & lp3 = mesh->Point (el.PNum(trigs[j][2]));
+		Vec3d n = Cross (Vec3d (lp1, lp2), Vec3d (lp1, lp3));
+		n /= (fac * (n.Length() + 1e-12));
+		glNormal3d (n.X(), n.Y(), n.Z());
+		glVertex3d (lp1.X(), lp1.Y(), lp1.Z());
+		glVertex3d (lp2.X(), lp2.Y(), lp2.Z());
+		glVertex3d (lp3.X(), lp3.Y(), lp3.Z());
+	      }
+	    glEnd();
+	  }
+      }
+    glEndList ();
+  }
 
 
 
@@ -2558,171 +2931,171 @@ void VisualSceneMesh :: BuildDomainSurfList()
 
 
 
-void VisualSceneMesh :: MouseDblClick (int px, int py)
-{
-  int i, hits;
+  void VisualSceneMesh :: MouseDblClick (int px, int py)
+  {
+    int i, hits;
 
-  // select surface triangle by mouse click
+    // select surface triangle by mouse click
 
-  GLuint selbuf[10000];
-  glSelectBuffer (10000, selbuf);
+    GLuint selbuf[10000];
+    glSelectBuffer (10000, selbuf);
 
 
-  glRenderMode (GL_SELECT);
+    glRenderMode (GL_SELECT);
 
-  GLint viewport[4];
-  glGetIntegerv (GL_VIEWPORT, viewport);
+    GLint viewport[4];
+    glGetIntegerv (GL_VIEWPORT, viewport);
 
 
-  glMatrixMode (GL_PROJECTION); 
-  glPushMatrix();
+    glMatrixMode (GL_PROJECTION); 
+    glPushMatrix();
 
-  GLdouble projmat[16];
-  glGetDoublev (GL_PROJECTION_MATRIX, projmat);
+    GLdouble projmat[16];
+    glGetDoublev (GL_PROJECTION_MATRIX, projmat);
 
-  glLoadIdentity(); 
-  gluPickMatrix (px, viewport[3] - py, 1, 1, viewport); 
-  glMultMatrixd (projmat);
+    glLoadIdentity(); 
+    gluPickMatrix (px, viewport[3] - py, 1, 1, viewport); 
+    glMultMatrixd (projmat);
   
 
 
-  glClearColor(backcolor, backcolor, backcolor, 1.0);
-  glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
+    glClearColor(backcolor, backcolor, backcolor, 1.0);
+    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
 
-  glMatrixMode (GL_MODELVIEW); 
+    glMatrixMode (GL_MODELVIEW); 
 
-  glPushMatrix();
-  glMultMatrixf (transformationmat);
+    glPushMatrix();
+    glMultMatrixf (transformationmat);
 
 
-  //  SetClippingPlane();
+    //  SetClippingPlane();
 
-  glInitNames();
-  glPushName (1);
+    glInitNames();
+    glPushName (1);
 
-  glPolygonOffset (1, 1);
-  glEnable (GL_POLYGON_OFFSET_FILL);
+    glPolygonOffset (1, 1);
+    glEnable (GL_POLYGON_OFFSET_FILL);
 
-  glDisable(GL_CLIP_PLANE0);
+    glDisable(GL_CLIP_PLANE0);
   
-  if (vispar.clipenable)
-    {
-      Vec<3> n(clipplane[0], clipplane[1], clipplane[2]);
-      double len = Abs(n);
-      double mu = -clipplane[3] / (len*len);
-      Point<3> p (mu * n);
-      n /= len;
-      Vec<3> t1 = n.GetNormal ();
-      Vec<3> t2 = Cross (n, t1);
+    if (vispar.clipenable)
+      {
+	Vec<3> n(clipplane[0], clipplane[1], clipplane[2]);
+	double len = Abs(n);
+	double mu = -clipplane[3] / (len*len);
+	Point<3> p (mu * n);
+	n /= len;
+	Vec<3> t1 = n.GetNormal ();
+	Vec<3> t2 = Cross (n, t1);
       
-      double xi1mid = (center - p) * t1;
-      double xi2mid = (center - p) * t2;
+	double xi1mid = (center - p) * t1;
+	double xi2mid = (center - p) * t2;
       
-      glLoadName (0);  
-      glBegin (GL_QUADS);
-      glVertex3dv (p + (xi1mid-rad) * t1 + (xi2mid-rad) * t2);
-      glVertex3dv (p + (xi1mid+rad) * t1 + (xi2mid-rad) * t2);
-      glVertex3dv (p + (xi1mid+rad) * t1 + (xi2mid+rad) * t2);
-      glVertex3dv (p + (xi1mid-rad) * t1 + (xi2mid+rad) * t2);
-      glEnd ();
-    }
+	glLoadName (0);  
+	glBegin (GL_QUADS);
+	glVertex3dv (p + (xi1mid-rad) * t1 + (xi2mid-rad) * t2);
+	glVertex3dv (p + (xi1mid+rad) * t1 + (xi2mid-rad) * t2);
+	glVertex3dv (p + (xi1mid+rad) * t1 + (xi2mid+rad) * t2);
+	glVertex3dv (p + (xi1mid-rad) * t1 + (xi2mid+rad) * t2);
+	glEnd ();
+      }
 
-  //  SetClippingPlane();
+    //  SetClippingPlane();
 
-  glCallList (filledlist);
+    glCallList (filledlist);
 
-  glDisable (GL_POLYGON_OFFSET_FILL);
+    glDisable (GL_POLYGON_OFFSET_FILL);
   
-  glPopName();
+    glPopName();
 
-  glMatrixMode (GL_PROJECTION); 
-  glPopMatrix();
+    glMatrixMode (GL_PROJECTION); 
+    glPopMatrix();
 
-  glMatrixMode (GL_MODELVIEW); 
-  glPopMatrix();
+    glMatrixMode (GL_MODELVIEW); 
+    glPopMatrix();
 
-  glFlush();  
+    glFlush();  
 
 	
-  hits = glRenderMode (GL_RENDER);
-
-  //  cout << "hits = " << hits << endl;
-
-  int minname = 0;
-  GLuint mindepth = 0;
-
-  // find clippingplane
-  GLuint clipdepth = 0; // GLuint(-1);
-
-  for (i = 0; i < hits; i++)
-    {
-      int curname = selbuf[4*i+3];
-      if (!curname) clipdepth = selbuf[4*i+1];
-    }
-
-  for (i = 0; i < hits; i++)
-    {
-      int curname = selbuf[4*i+3];
-      GLuint curdepth = selbuf[4*i+1];
-      /*
-      cout << selbuf[4*i] << " " << selbuf[4*i+1] << " " 
-	   << selbuf[4*i+2] << " " << selbuf[4*i+3] << endl;
-      */
-      if (curname && (curdepth > clipdepth) &&
-	  (curdepth < mindepth || !minname))
-	{
-	  mindepth = curdepth;
-	  minname = curname;
-	}
-    }
-
-  seledge = -1;
-  if (minname)
-    {
-      const Element2d & sel = mesh->SurfaceElement(minname);
-
-
-      cout << "select element " << minname
-	   << " on face " << sel.GetIndex() << endl;
-      cout << "Nodes: ";
-      for (i = 1; i <= sel.GetNP(); i++)
-	cout << sel.PNum(i) << " ";
-      cout << endl;
-
-      selelement = minname;
-      selface = mesh->SurfaceElement(minname).GetIndex();
-
-      locpi = (locpi % sel.GetNP()) + 1;
-      selpoint2 = selpoint;
-      selpoint = sel.PNum(locpi);
-      cout << "selected point " << selpoint 
-	   << ", pos = " << mesh->Point (selpoint) 
-	   << endl;
-
-      for (i = 1; i <= mesh->GetNSeg(); i++)
-	{
-	  const Segment & seg = mesh->LineSegment(i);
-	  if (seg.p1 == selpoint && seg.p2 == selpoint2 || 
-	      seg.p2 == selpoint && seg.p1 == selpoint2)
-	    {
-	      seledge = seg.edgenr;
-	      cout << "seledge = " << seledge << endl;
-	    }
-	}
+    hits = glRenderMode (GL_RENDER);
+
+    //  cout << "hits = " << hits << endl;
+
+    int minname = 0;
+    GLuint mindepth = 0;
+
+    // find clippingplane
+    GLuint clipdepth = 0; // GLuint(-1);
+
+    for (i = 0; i < hits; i++)
+      {
+	int curname = selbuf[4*i+3];
+	if (!curname) clipdepth = selbuf[4*i+1];
+      }
+
+    for (i = 0; i < hits; i++)
+      {
+	int curname = selbuf[4*i+3];
+	GLuint curdepth = selbuf[4*i+1];
+	/*
+	  cout << selbuf[4*i] << " " << selbuf[4*i+1] << " " 
+	  << selbuf[4*i+2] << " " << selbuf[4*i+3] << endl;
+	*/
+	if (curname && (curdepth > clipdepth) &&
+	    (curdepth < mindepth || !minname))
+	  {
+	    mindepth = curdepth;
+	    minname = curname;
+	  }
+      }
+
+    seledge = -1;
+    if (minname)
+      {
+	const Element2d & sel = mesh->SurfaceElement(minname);
+
+
+	cout << "select element " << minname
+	     << " on face " << sel.GetIndex() << endl;
+	cout << "Nodes: ";
+	for (i = 1; i <= sel.GetNP(); i++)
+	  cout << sel.PNum(i) << " ";
+	cout << endl;
+
+	selelement = minname;
+	selface = mesh->SurfaceElement(minname).GetIndex();
+
+	locpi = (locpi % sel.GetNP()) + 1;
+	selpoint2 = selpoint;
+	selpoint = sel.PNum(locpi);
+	cout << "selected point " << selpoint 
+	     << ", pos = " << mesh->Point (selpoint) 
+	     << endl;
+
+	for (i = 1; i <= mesh->GetNSeg(); i++)
+	  {
+	    const Segment & seg = mesh->LineSegment(i);
+	    if (seg.p1 == selpoint && seg.p2 == selpoint2 || 
+		seg.p2 == selpoint && seg.p1 == selpoint2)
+	      {
+		seledge = seg.edgenr;
+		cout << "seledge = " << seledge << endl;
+	      }
+	  }
       
-    }
-  else
-    {
-      selface = -1;
-      selelement = -1;
-      selpoint = -1;
-      selpoint2 = -1;
-    }
-
-  glDisable(GL_CLIP_PLANE0);
-
-  selecttimestamp = NextTimeStamp();
-}
+      }
+    else
+      {
+	selface = -1;
+	selelement = -1;
+	selpoint = -1;
+	selpoint2 = -1;
+      }
+
+    glDisable(GL_CLIP_PLANE0);
+
+    selecttimestamp = NextTimeStamp();
+  }
 
 
 
diff --git a/Netgen/libsrc/visualization/vsocc.cpp b/Netgen/libsrc/visualization/vsocc.cpp
new file mode 100644
index 0000000000..12cfc5e179
--- /dev/null
+++ b/Netgen/libsrc/visualization/vsocc.cpp
@@ -0,0 +1,743 @@
+#ifdef OCCGEOMETRY
+
+
+#include <mystdlib.h>
+#include <myadt.hpp>
+#include <meshing.hpp>
+
+// #include <csg.hpp>
+// #include <stlgeom.hpp>
+
+#include <occgeom.hpp>
+
+#include "TopoDS_Shape.hxx"
+#include "TopoDS_Vertex.hxx"
+#include "TopExp_Explorer.hxx"
+#include "BRep_Tool.hxx"
+#include "TopoDS.hxx"
+#include "gp_Pnt.hxx"
+#include "Geom_Curve.hxx"
+#include "Poly_Triangulation.hxx"
+#include "Poly_Array1OfTriangle.hxx"
+#include "TColgp_Array1OfPnt2d.hxx"
+#include "Poly_Triangle.hxx"
+#include "Poly_Polygon3D.hxx"
+#include "Poly_PolygonOnTriangulation.hxx"
+// #include "BRepMesh.hxx"
+// #include "BRepMesh_IncrementalMesh.hxx"
+
+#include "incvis.hpp"
+
+
+namespace netgen
+{
+#include "mvdraw.hpp"
+
+
+extern OCCGeometry * occgeometry;
+
+
+
+
+/* *********************** Draw OCC Geometry **************** */
+
+
+VisualSceneOCCGeometry :: VisualSceneOCCGeometry ()
+  : VisualScene()
+{
+  trilists.SetSize(0);
+  linelists.SetSize(1);
+
+}
+
+VisualSceneOCCGeometry :: ~VisualSceneOCCGeometry ()
+{
+  ;
+}
+
+void VisualSceneOCCGeometry :: DrawScene ()
+{
+  // int i, j, k;
+
+  if ( occgeometry->changed )
+    {
+      BuildScene();
+      occgeometry -> changed = 0;
+    }
+
+  glClearColor(backcolor, backcolor, backcolor, 1.0);
+  glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
+  SetLight();
+
+  glPushMatrix();
+  glMultMatrixf (transformationmat);
+
+  glShadeModel (GL_SMOOTH);
+  glDisable (GL_COLOR_MATERIAL);
+  glPolygonMode (GL_FRONT_AND_BACK, GL_FILL);
+
+  glEnable (GL_BLEND);
+  glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
+
+  //  glEnable (GL_LIGHTING);
+
+  double shine = vispar.shininess;
+  // double transp = vispar.transp;
+
+  glMaterialf (GL_FRONT_AND_BACK, GL_SHININESS, shine);
+  glLogicOp (GL_COPY);
+
+
+  float mat_col[] = { 0.2f, 0.2f, 0.8f, 1.0f };
+  glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col);
+
+  glPolygonOffset (1, 1);
+  glEnable (GL_POLYGON_OFFSET_FILL);
+
+  GLfloat matcoledge[] = { 0, 0, 1, 1 };
+  GLfloat matcolhiedge[] = { 1, 0, 0, 1 };
+
+  glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, 
+		matcoledge);
+  glLineWidth (1.0f);
+
+  if (vispar.occshowedges) glCallList (linelists.Get(1));
+  if (vispar.occshowsurfaces) glCallList (trilists.Get(1));
+
+  glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, 
+		matcolhiedge);
+  glLineWidth (5.0f);
+
+  if (vispar.occshowedges) glCallList (linelists.Get(2));
+
+  for (int i = 1; i <= occgeometry->vmap.Extent(); i++)
+    if (occgeometry->vvispar[i-1].IsHighlighted())
+      {
+	glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, 
+		      matcolhiedge);
+	glLineWidth (5.0f);
+
+	glBegin (GL_LINES);
+
+	gp_Pnt p = BRep_Tool::Pnt(TopoDS::Vertex(occgeometry->vmap(i)));
+	double d = rad/100;
+	glVertex3f (p.X()-d, p.Y(), p.Z());
+	glVertex3f (p.X()+d, p.Y(), p.Z());
+	glVertex3f (p.X(), p.Y()-d, p.Z());
+	glVertex3f (p.X(), p.Y()+d, p.Z());
+	glVertex3f (p.X(), p.Y(), p.Z()-d);
+	glVertex3f (p.X(), p.Y(), p.Z()+d);
+	glEnd();
+      }
+
+  glDisable (GL_POLYGON_OFFSET_FILL);
+ 
+  glPopMatrix();
+  //  DrawCoordinateCross ();
+  //  DrawNetgenLogo ();
+  glFinish();  
+
+  glDisable (GL_POLYGON_OFFSET_FILL);
+}
+
+
+/*
+void VisualSceneOCCGeometry :: BuildScene (int zoomall)
+{
+  int i = 0, j, k;
+  
+  TopExp_Explorer ex, ex_edge;
+
+  if (vispar.occvisproblemfaces || (occgeometry -> changed != 2))
+    {
+      Box<3> bb = occgeometry -> GetBoundingBox();
+
+      center = bb.Center();
+      rad = bb.Diam() / 2;
+  
+
+
+      if (vispar.occvisproblemfaces)
+	{
+	  for (i = 1; i <= occgeometry->fmap.Extent(); i++)
+	    if (occgeometry->facemeshstatus[i-1] == -1)
+	      {
+		GProp_GProps system;	       
+		BRepGProp::LinearProperties(occgeometry->fmap(i), system);
+		gp_Pnt pnt = system.CentreOfMass();
+		center = Point<3> (pnt.X(), pnt.Y(), pnt.Z());
+		cout << "Setting center to mid of face " << i << " = " << center << endl;
+	      }
+	}
+
+
+      CalcTransformationMatrices();
+    }
+  
+
+  for (i = 1; i <= linelists.Size(); i++)
+    glDeleteLists (linelists.Elem(i), 1);
+  linelists.SetSize(0);
+
+  linelists.Append (glGenLists (1));
+  glNewList (linelists.Last(), GL_COMPILE);
+  
+  i = 0;
+  for (ex_edge.Init(occgeometry -> shape, TopAbs_EDGE);
+       ex_edge.More(); ex_edge.Next())
+    {
+      if (BRep_Tool::Degenerated(TopoDS::Edge(ex_edge.Current()))) continue;
+      i++;
+
+      
+      TopoDS_Edge edge = TopoDS::Edge(ex_edge.Current());
+
+      Handle(Poly_PolygonOnTriangulation) aEdgePoly;
+      Handle(Poly_Triangulation) T;
+      TopLoc_Location aEdgeLoc;
+      BRep_Tool::PolygonOnTriangulation(edge, aEdgePoly, T, aEdgeLoc);
+     
+      if(aEdgePoly.IsNull())
+	{
+	  cout << "cannot visualize edge " << i << endl;
+	  continue;
+	}
+
+      glBegin (GL_LINE_STRIP);
+	  
+      int nbnodes = aEdgePoly -> NbNodes();
+      for (j = 1; j <= nbnodes; j++)
+	{
+	  gp_Pnt p = (T -> Nodes())(aEdgePoly->Nodes()(j)).Transformed(aEdgeLoc);
+	  glVertex3f (p.X(), p.Y(), p.Z());
+	}
+      
+      glEnd ();
+      
+   
+      }
+  
+  glEndList ();
+
+  for (i = 1; i <= trilists.Size(); i++)
+    glDeleteLists (trilists.Elem(i), 1);
+  trilists.SetSize(0);
+
+
+  trilists.Append (glGenLists (1));
+  glNewList (trilists.Last(), GL_COMPILE);
+
+  i = 0;
+
+  TopExp_Explorer exp0, exp1, exp2, exp3;
+  int shapenr = 0;
+  for (exp0.Init(occgeometry -> shape, TopAbs_SOLID); exp0.More(); exp0.Next())
+    {
+      shapenr++;
+
+      if (vispar.occshowvolumenr != 0 && 
+	  vispar.occshowvolumenr != shapenr) continue;
+
+      float mat_col[4];
+      mat_col[3] = 1;
+      switch (shapenr)
+	{
+	case 1:
+	  mat_col[0] = 0.2;
+	  mat_col[1] = 0.2;
+	  mat_col[2] = 0.8;
+	  break;
+	case 2:
+	  mat_col[0] = 0.8;
+	  mat_col[1] = 0.2;
+	  mat_col[2] = 0.8;
+	  break;
+	case 3:
+	  mat_col[0] = 0.2;
+	  mat_col[1] = 0.8;
+	  mat_col[2] = 0.8;
+	  break;
+	case 4:
+	  mat_col[0] = 0.8;
+	  mat_col[1] = 0.2;
+	  mat_col[2] = 0.2;
+	  break;
+	case 5:
+	  mat_col[0] = 0.8;
+	  mat_col[1] = 0.8;
+	  mat_col[2] = 0.8;
+	  break;
+	case 6:
+	  mat_col[0] = 0.6;
+	  mat_col[1] = 0.6;
+	  mat_col[2] = 0.6;
+	  break;
+	case 7:
+	  mat_col[0] = 0.2;
+	  mat_col[1] = 0.8;
+	  mat_col[2] = 0.2;
+	  break;
+	case 8:
+	  mat_col[0] = 0.8;
+	  mat_col[1] = 0.8;
+	  mat_col[2] = 0.2;
+	  break;
+	default:
+	  //	  mat_col[0] = 1-(1.0/double(shapenr));
+	  //	  mat_col[1] = 0.5;
+	  mat_col[0] = 0.5+double((shapenr*shapenr*shapenr*shapenr) % 10)/20.0;
+	  mat_col[1] = 0.5+double(int(shapenr*shapenr*shapenr*shapenr*sin(double(shapenr))) % 10)/20.0;
+	  mat_col[2] = 0.5+double((shapenr*shapenr*shapenr) % 10)/20.0;
+	}
+
+      glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col);
+
+      for (exp1.Init(exp0.Current(), TopAbs_SHELL); exp1.More(); exp1.Next())
+	for (exp2.Init(exp1.Current().Composed(exp0.Current().Orientation()), TopAbs_FACE); exp2.More(); exp2.Next())
+	  {
+	    TopoDS_Face face = TopoDS::Face (exp2.Current().Composed(exp1.Current().Orientation()));
+	    
+	    i = occgeometry->fmap.FindIndex(face);
+
+	    TopLoc_Location loc;
+	    Handle(Geom_Surface) surf = BRep_Tool::Surface (face);
+	    BRepAdaptor_Surface sf(face, Standard_False);
+	    BRepLProp_SLProps prop(sf, 1, 1e-5);
+	    Handle(Poly_Triangulation) triangulation = BRep_Tool::Triangulation (face, loc);
+	    
+	    if (triangulation.IsNull()) 
+	      {
+		cout << "cannot visualize face " << i << endl;
+		continue;
+	      }
+
+	    if (vispar.occvisproblemfaces)
+	      {
+		switch (occgeometry->facemeshstatus[i-1])
+		  {
+		  case 0:
+		    mat_col[0] = 0.2;
+		    mat_col[1] = 0.2;
+		    mat_col[2] = 0.8;
+		    break;		
+		  case 1:
+		    mat_col[0] = 0.2;
+		    mat_col[1] = 0.8;
+		    mat_col[2] = 0.2;
+		    break;		
+		  case -1:
+		    mat_col[0] = 0.8;
+		    mat_col[1] = 0.2;
+		    mat_col[2] = 0.2;
+		    break;		
+		  }
+		glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col);
+
+      	      }
+	    glBegin (GL_TRIANGLES);
+	    
+	    int ntriangles = triangulation -> NbTriangles();
+	    for (j = 1; j <= ntriangles; j++)
+	      {
+		Poly_Triangle triangle = (triangulation -> Triangles())(j);
+		for (k = 1; k <= 3; k++)
+		  {
+		    gp_Pnt2d uv = (triangulation -> UVNodes())(triangle(k));
+		    gp_Pnt pnt;
+		    gp_Vec du, dv;
+		    prop.SetParameters (uv.X(), uv.Y());
+		    surf->D0 (uv.X(), uv.Y(), pnt);
+		    gp_Vec n;
+		    
+		    if (prop.IsNormalDefined())
+		      n = prop.Normal();
+		    else
+		      n = gp_Vec (0,0,0);
+		    
+		    if (face.Orientation() == TopAbs_REVERSED) n *= -1;
+		    glNormal3f (n.X(), n.Y(), n.Z());
+		    glVertex3f (pnt.X(), pnt.Y(), pnt.Z());
+		  }
+	      }
+	    glEnd ();
+
+	  }
+    }
+
+
+  glEndList ();
+
+}
+*/
+
+
+void VisualSceneOCCGeometry :: BuildScene (int zoomall)
+{
+  if (occgeometry -> changed == OCCGEOMETRYVISUALIZATIONFULLCHANGE)
+    {
+      center = occgeometry -> Center();
+      rad = occgeometry -> GetBoundingBox().Diam() / 2;
+
+      if (vispar.occzoomtohighlightedentity)
+	{
+	  bool hilite = false;
+	  bool hiliteonepoint = false;
+	  Bnd_Box bb;
+
+	  for (int i = 1; i <= occgeometry->fmap.Extent(); i++)
+	    if (occgeometry->fvispar[i-1].IsHighlighted())
+	      {
+		hilite = true;
+		BRepBndLib::Add (occgeometry->fmap(i), bb);
+	      }
+
+	  for (int i = 1; i <= occgeometry->emap.Extent(); i++)
+	    if (occgeometry->evispar[i-1].IsHighlighted())
+	      {
+		hilite = true;
+		BRepBndLib::Add (occgeometry->emap(i), bb);
+	      }
+
+	  for (int i = 1; i <= occgeometry->vmap.Extent(); i++)
+	    if (occgeometry->vvispar[i-1].IsHighlighted())
+	      {
+		hiliteonepoint = true;
+		BRepBndLib::Add (occgeometry->vmap(i), bb);
+	      }
+
+	  if (hilite || hiliteonepoint)
+	    {
+	      double x1,y1,z1,x2,y2,z2;
+	      bb.Get (x1,y1,z1,x2,y2,z2);
+	      Point<3> p1 = Point<3> (x1,y1,z1);
+	      Point<3> p2 = Point<3> (x2,y2,z2);
+	      Box<3> boundingbox(p1,p2);
+	      
+	      center = boundingbox.Center();
+	      if (hiliteonepoint)
+		rad = occgeometry -> GetBoundingBox().Diam() / 100; 
+	      else
+		rad = boundingbox.Diam() / 2;
+	    }
+	}
+  
+      CalcTransformationMatrices();
+    }
+  
+
+  // Clear lists
+
+  for (int i = 1; i <= linelists.Size(); i++)
+    glDeleteLists (linelists.Elem(i), 1);
+  linelists.SetSize(0);
+
+  for (int i = 1; i <= trilists.Size(); i++)
+    glDeleteLists (trilists.Elem(i), 1);
+  trilists.SetSize(0);
+
+
+  // Total wireframe
+
+  linelists.Append (glGenLists (1));
+  glNewList (linelists.Last(), GL_COMPILE);
+ 
+  for (int i = 1; i <= occgeometry->emap.Extent(); i++)
+    {
+      TopoDS_Edge edge = TopoDS::Edge(occgeometry->emap(i));
+      if (BRep_Tool::Degenerated(edge)) continue;
+      if (occgeometry->evispar[i-1].IsHighlighted()) continue;
+
+      Handle(Poly_PolygonOnTriangulation) aEdgePoly;
+      Handle(Poly_Triangulation) T;
+      TopLoc_Location aEdgeLoc;
+      BRep_Tool::PolygonOnTriangulation(edge, aEdgePoly, T, aEdgeLoc);
+     
+      if(aEdgePoly.IsNull())
+	{
+	  (*testout) << "visualizing edge " << occgeometry->emap.FindIndex (edge)
+		     << " without using the occ visualization triangulation" << endl;
+
+	  double s0, s1;
+	  Handle(Geom_Curve) c = BRep_Tool::Curve(edge, s0, s1);
+	  
+	  glBegin (GL_LINE_STRIP);
+	  for (int i = 0; i<=50; i++)
+	    {
+	      gp_Pnt p = c->Value (s0 + i*(s1-s0)/50.0);
+	      glVertex3f (p.X(),p.Y(),p.Z());
+	    }
+	  glEnd ();
+
+	  continue;
+	}
+
+      int nbnodes = aEdgePoly -> NbNodes();
+      glBegin (GL_LINE_STRIP);
+      for (int j = 1; j <= nbnodes; j++)
+	{
+	  gp_Pnt p = (T -> Nodes())(aEdgePoly->Nodes()(j)).Transformed(aEdgeLoc);
+	  glVertex3f (p.X(), p.Y(), p.Z());
+	}
+      glEnd ();
+    }
+  
+  glEndList ();
+
+
+  // Highlighted edge list
+
+  linelists.Append (glGenLists (1));
+  glNewList (linelists.Last(), GL_COMPILE);
+ 
+  for (int i = 1; i <= occgeometry->emap.Extent(); i++)
+    if (occgeometry->evispar[i-1].IsHighlighted())
+      {
+	TopoDS_Edge edge = TopoDS::Edge(occgeometry->emap(i));
+	if (BRep_Tool::Degenerated(edge)) continue;
+
+	Handle(Poly_PolygonOnTriangulation) aEdgePoly;
+	Handle(Poly_Triangulation) T;
+	TopLoc_Location aEdgeLoc;
+	BRep_Tool::PolygonOnTriangulation(edge, aEdgePoly, T, aEdgeLoc);
+     
+	if(aEdgePoly.IsNull())
+	  {
+	    (*testout) << "visualizing edge " << occgeometry->emap.FindIndex (edge)
+		       << " without using the occ visualization triangulation" << endl;
+
+	    double s0, s1;
+	    Handle(Geom_Curve) c = BRep_Tool::Curve(edge, s0, s1);
+	    
+	    glBegin (GL_LINE_STRIP);
+	    for (int i = 0; i<=50; i++)
+	      {
+		gp_Pnt p = c->Value (s0 + i*(s1-s0)/50.0);
+		glVertex3f (p.X(),p.Y(),p.Z());
+	      }
+	    glEnd ();
+	    
+	    continue;
+	  }
+
+	int nbnodes = aEdgePoly -> NbNodes();
+	glBegin (GL_LINE_STRIP);
+	for (int j = 1; j <= nbnodes; j++)
+	  {
+	    gp_Pnt p = (T -> Nodes())(aEdgePoly->Nodes()(j)).Transformed(aEdgeLoc);
+	    glVertex3f (p.X(), p.Y(), p.Z());
+	  }
+	glEnd ();
+      }
+  
+  glEndList ();
+
+
+
+
+
+  // display faces
+
+  trilists.Append (glGenLists (1));
+  glNewList (trilists.Last(), GL_COMPILE);
+
+  for (int i = 1; i <= occgeometry->fmap.Extent(); i++)
+    {
+      glLoadName (i);
+      float mat_col[4];
+      mat_col[3] = 1;
+
+      if (!occgeometry->fvispar[i-1].IsHighlighted())
+	{
+	  mat_col[0] = 0.2;
+	  mat_col[1] = 0.2;
+	  mat_col[2] = 0.8;
+	}
+      else
+	{
+	  mat_col[0] = 0.8;
+	  mat_col[1] = 0.2;
+	  mat_col[2] = 0.2;
+	}
+
+      glMaterialfv (GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, mat_col);
+
+      TopoDS_Face face = TopoDS::Face(occgeometry->fmap(i));
+      TopLoc_Location loc;
+      Handle(Geom_Surface) surf = BRep_Tool::Surface (face);
+      BRepAdaptor_Surface sf(face, Standard_False);
+      BRepLProp_SLProps prop(sf, 1, 1e-5);
+      Handle(Poly_Triangulation) triangulation = BRep_Tool::Triangulation (face, loc);
+	    
+      if (triangulation.IsNull()) 
+	{
+	  cout << "cannot visualize face " << i << endl;
+	  occgeometry->fvispar[i-1].SetNotDrawable();
+	  continue;
+	}
+      
+      gp_Pnt2d uv;
+      gp_Pnt pnt;
+      gp_Vec n;
+
+      glBegin (GL_TRIANGLES);
+	    
+      int ntriangles = triangulation -> NbTriangles();
+      for (int j = 1; j <= ntriangles; j++)
+	{
+	  Poly_Triangle triangle = (triangulation -> Triangles())(j);
+	  for (int k = 1; k <= 3; k++)
+	    {
+	      uv = (triangulation -> UVNodes())(triangle(k));
+	      prop.SetParameters (uv.X(), uv.Y());
+
+	      pnt = (triangulation -> Nodes())(triangle(k)).Transformed(loc);
+
+	      //	      surf->D0 (uv.X(), uv.Y(), pnt);
+	      		    
+	      if (prop.IsNormalDefined())
+		n = prop.Normal();
+	      else
+		{
+		  (*testout) << "Visualization of face " << i
+			     << ": Normal vector not defined" << endl;
+		  n = gp_Vec (0,0,0);
+		}
+		    
+	      if (face.Orientation() == TopAbs_REVERSED) n *= -1;
+	      glNormal3f (n.X(), n.Y(), n.Z());
+	      glVertex3f (pnt.X(), pnt.Y(), pnt.Z());
+	    }
+	}
+      glEnd ();
+      
+    }
+  glEndList ();
+
+}
+
+void SelectFaceInOCCDialogTree (int facenr);
+
+void VisualSceneOCCGeometry :: MouseDblClick (int px, int py)
+{
+  int hits;
+
+  // select surface triangle by mouse click
+
+  GLuint selbuf[10000];
+  glSelectBuffer (10000, selbuf);
+
+
+  glRenderMode (GL_SELECT);
+
+  GLint viewport[4];
+  glGetIntegerv (GL_VIEWPORT, viewport);
+
+
+  glMatrixMode (GL_PROJECTION); 
+  glPushMatrix();
+
+  GLdouble projmat[16];
+  glGetDoublev (GL_PROJECTION_MATRIX, projmat);
+
+  glLoadIdentity(); 
+  gluPickMatrix (px, viewport[3] - py, 1, 1, viewport); 
+  glMultMatrixd (projmat);
+  
+
+
+  glClearColor(backcolor, backcolor, backcolor, 1.0);
+  glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
+
+  glMatrixMode (GL_MODELVIEW); 
+
+  glPushMatrix();
+  glMultMatrixf (transformationmat);
+
+
+
+  glInitNames();
+  glPushName (1);
+
+  glPolygonOffset (1, 1);
+  glEnable (GL_POLYGON_OFFSET_FILL);
+
+  glDisable(GL_CLIP_PLANE0);
+  
+  glCallList (trilists.Get(1));
+
+  glDisable (GL_POLYGON_OFFSET_FILL);
+  
+  glPopName();
+
+  glMatrixMode (GL_PROJECTION); 
+  glPopMatrix();
+
+  glMatrixMode (GL_MODELVIEW); 
+  glPopMatrix();
+
+  glFlush();  
+
+	
+  hits = glRenderMode (GL_RENDER);
+
+  int minname = 0;
+  GLuint mindepth = 0;
+
+  // find clippingplane
+  GLuint clipdepth = 0; // GLuint(-1);
+
+  for (int i = 0; i < hits; i++)
+    {
+      int curname = selbuf[4*i+3];
+      if (!curname) clipdepth = selbuf[4*i+1];
+    }
+
+  for (int i = 0; i < hits; i++)
+    {
+      int curname = selbuf[4*i+3];
+      GLuint curdepth = selbuf[4*i+1];
+      if (curname && (curdepth > clipdepth) &&
+	  (curdepth < mindepth || !minname))
+	{
+	  mindepth = curdepth;
+	  minname = curname;
+	}
+    }
+
+  occgeometry->LowLightAll();
+
+  if (minname)
+    {
+      occgeometry->fvispar[minname-1].Highlight();
+      
+      if (vispar.occzoomtohighlightedentity)
+	occgeometry->changed = OCCGEOMETRYVISUALIZATIONFULLCHANGE;
+      else
+	occgeometry->changed = OCCGEOMETRYVISUALIZATIONHALFCHANGE;
+      cout << "Selected face: " << minname << endl;
+    }
+  else
+    {
+      occgeometry->changed = OCCGEOMETRYVISUALIZATIONHALFCHANGE;
+    }
+
+  glDisable(GL_CLIP_PLANE0);
+
+  SelectFaceInOCCDialogTree (minname);
+
+
+  //  selecttimestamp = NextTimeStamp();
+}
+
+
+
+
+
+
+}
+
+
+
+#endif
+
+
diff --git a/Netgen/libsrc/visualization/vssolution.cpp b/Netgen/libsrc/visualization/vssolution.cpp
index 8678fc900d..62b6faab83 100644
--- a/Netgen/libsrc/visualization/vssolution.cpp
+++ b/Netgen/libsrc/visualization/vssolution.cpp
@@ -15,6 +15,18 @@ namespace netgen
 
   extern AutoPtr<Mesh> mesh;
 
+
+  VisualSceneSolution :: SolData :: SolData ()
+    : name (0), data (0), solclass(0)
+  { ; }
+
+  VisualSceneSolution :: SolData :: ~SolData ()
+  {
+    delete [] name;
+    delete data;
+    delete solclass;
+  }
+
   
   VisualSceneSolution :: VisualSceneSolution ()
     : VisualScene()
@@ -23,10 +35,12 @@ namespace netgen
     linelist = 0;
     clipplanelist = 0;
     isolinelist = 0;
+    clipplane_isolinelist = 0;
     surface_vector_list = 0;
     cone_list = 0;
 
     surfeltimestamp = GetTimeStamp();
+    surfellinetimestamp = GetTimeStamp();
     clipplanetimestamp = GetTimeStamp();
     solutiontimestamp = GetTimeStamp();
     fieldlinestimestamp = GetTimeStamp();
@@ -36,22 +50,23 @@ namespace netgen
   
   VisualSceneSolution :: ~VisualSceneSolution ()
   {
-    ;
+    ClearSolutionData();
   }
 
   void VisualSceneSolution :: AddSolutionData (SolData * sd)
   {
-    int i, funcnr = -1;
-    
-    for (i = 0; i < soldata.Size(); i++)
+    int funcnr = -1;
+    for (int i = 0; i < soldata.Size(); i++)
       {
 	if (strcmp (soldata[i]->name, sd->name) == 0)
 	  {
+	    delete soldata[i];
 	    soldata[i] = sd;
 	    funcnr = i;
 	    break;
 	  }
       }
+
     if (funcnr == -1)
       {
 	soldata.Append (sd);
@@ -91,18 +106,13 @@ namespace netgen
 	  }
 	solutiontimestamp = NextTimeStamp();
       }
-    
-    /*
-      (*testout) << "set sol-data " << nsd->name
-      << ", size = " << nsd->size << endl;
-      if (nsd->data)
-      for (i = 0; i < nsd->size; i++)
-      (*testout) << i << ": " << nsd->data[i] << endl;
-    */
   }
+
   
   void VisualSceneSolution :: ClearSolutionData ()
   {
+    for (int i = 0; i < soldata.Size(); i++)
+      delete soldata[i];
     soldata.SetSize (0);
   }
 
@@ -110,8 +120,6 @@ namespace netgen
   {
     solutiontimestamp = NextTimeStamp();
   }
-  
-
     
   VisualSceneSolution::SolData * VisualSceneSolution :: GetSolData (int i)
   { 
@@ -124,7 +132,7 @@ namespace netgen
 
 
 
-  void VisualSceneSolution :: SaveSolutionData (const char * filename)
+  void VisualSceneSolution :: SaveSolutionData (const char * filename) 
   {
     PrintMessage (1, "Write solution data to file ", filename);
     int i, j, k;
@@ -171,15 +179,6 @@ namespace netgen
 
   void VisualSceneSolution :: DrawScene ()
   {
-    /*
-    if (multithread.running) 
-      {
-	VisualScene::DrawScene();      
-	return;
-      }
-    */
-    int i, j, k;
-    
     if (!mesh) 
       {
 	VisualScene::DrawScene();      
@@ -189,24 +188,15 @@ namespace netgen
     static NgLock mem_lock(mem_mutex);
     mem_lock.Lock();
 
-    /*    
-    lock = NULL;
-    if (!lock)
-      {
-	lock = new NgLock (mesh->Mutex());
-	lock -> Lock();
-      }
-    */
+    NgLock meshlock (mesh->Mutex(), 1);
 
     BuildScene();
 
-    // either continuous, or discrete coloring
-    CreateTexture (numtexturecols, lineartexture);
+    CreateTexture (numtexturecols, lineartexture, GL_MODULATE);
 
-    glClearColor(backcolor, backcolor, backcolor, 1.0);
+    glClearColor(backcolor, backcolor, backcolor, 1);
     glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
 
-
     SetLight();
     
     glPushMatrix();
@@ -218,7 +208,8 @@ namespace netgen
     
     glPolygonMode (GL_FRONT_AND_BACK, GL_FILL);
     
-    glPolygonOffset (1, 1);
+    // glPolygonOffset (1, 1);
+    glPolygonOffset (2, 10);
     glEnable (GL_POLYGON_OFFSET_FILL);
 
     glEnable (GL_COLOR_MATERIAL);
@@ -228,7 +219,7 @@ namespace netgen
       glEnable (GL_TEXTURE_1D);
 
 
-    if (vispar.drawfilledtrigs)
+    if (vispar.drawfilledtrigs || vispar.drawtetsdomain > 0 || vispar.drawdomainsurf > 0)
       {
 	SetClippingPlane ();
 	
@@ -240,7 +231,7 @@ namespace netgen
 
     if (showclipsolution)
       glCallList (clipplanelist);
-  
+
 
     if (draw_fieldlines)
       {
@@ -249,7 +240,7 @@ namespace netgen
 	if (num_fieldlineslists <= 1)
 	  glCallList (fieldlineslist);
 	else
-	  {
+	  {  // animated
 	    int start = int (time / 10 * num_fieldlineslists);
 	    for (int ln = 0; ln < 10; ln++)
 	      {
@@ -278,43 +269,37 @@ namespace netgen
     glLineWidth (1.0f);
     glColor3f (0.0f, 0.0f, 0.0f);
     glDisable (GL_LINE_SMOOTH);
-    
-    if (vispar.drawoutline || numisolines)
+
+
+    if (vispar.drawoutline && !numisolines)
       {
 	SetClippingPlane ();
-	
-	if (numisolines)
-	  {
-	    glCallList (isolinelist);
-	  }
-	else
-	  {
-	    glCallList (linelist);
-	  }
-	
+	glCallList (linelist);
+       	glDisable(GL_CLIP_PLANE0);
+      }
+
+    if (numisolines)
+      {
+	SetClippingPlane ();
+	glCallList (isolinelist);
+
 	glDisable(GL_CLIP_PLANE0);
+	glCallList (clipplane_isolinelist);
       }
-    
-   
 
     glPopMatrix();
     
     glDisable(GL_CLIP_PLANE0);
-    DrawColorBar (minval, maxval, logscale);
+    DrawColorBar (minval, maxval, logscale, lineartexture);
     
     if (vispar.drawcoordinatecross)
       DrawCoordinateCross ();
     DrawNetgenLogo ();
     
     glFinish();  
+
     
-    /*
-    if (lock)
-      {
-	lock -> UnLock();
-	delete lock;
-      }
-    */
+    // delete lock;
     mem_lock.UnLock();
   }
   
@@ -384,8 +369,6 @@ namespace netgen
 	glEndList();
       }
 
-
-
     
     vispar.colormeshsize = 1;
     
@@ -394,20 +377,24 @@ namespace netgen
     glDisable(GL_CLIP_PLANE0);
     
     
-    const SolData * sol = NULL;
-    const SolData * vsol = NULL;
+    SolData * sol = NULL;
+    SolData * vsol = NULL;
   
     if (scalfunction != -1) 
       sol = soldata[scalfunction];
     if (vecfunction != -1)
       vsol = soldata[vecfunction];
-    
+
   if (mesh->GetTimeStamp () > solutiontimestamp)
     {
       sol = NULL;
       vsol = NULL;
     }
 
+
+  if (sol && sol->solclass) sol->solclass->SetMultiDimComponent (multidimcomponent);
+  if (vsol && vsol->solclass) vsol->solclass->SetMultiDimComponent (multidimcomponent);
+
   if (!autoscale || scalfunction == -1)
     {
       minval = mminval;
@@ -425,7 +412,7 @@ namespace netgen
  
 
   if (mesh->GetTimeStamp() > surfeltimestamp ||
-      solutiontimestamp > surfeltimestamp ||
+      solutiontimestamp > surfeltimestamp || 
       zoomall)
     {
       if (mesh->GetTimeStamp() > surfeltimestamp ||
@@ -451,18 +438,24 @@ namespace netgen
 	    }
 	}
 
-      
       if (surfellist)
 	glDeleteLists (surfellist, 1);
       
       surfellist = glGenLists (1);
       glNewList (surfellist, GL_COMPILE);
       
-      DrawSurfaceElements();
+	  DrawSurfaceElements();
       
       glEndList ();
       
-      
+      surfeltimestamp = max2 (solutiontimestamp, mesh->GetTimeStamp());
+    }
+
+
+  if (mesh->GetTimeStamp() > surfellinetimestamp ||
+      solutiontimestamp > surfellinetimestamp || 
+      zoomall)
+    {
       if (linelist)
 	glDeleteLists (linelist, 1);
       
@@ -472,15 +465,12 @@ namespace netgen
       DrawSurfaceElementLines();
       
       glEndList ();
-
-      surfeltimestamp = max2 (solutiontimestamp, mesh->GetTimeStamp());
-
-      //      cout << "minval = " << minval << " maxval = " << maxval << endl;
+      
+      surfellinetimestamp = max2 (solutiontimestamp, mesh->GetTimeStamp());
     }
-  
-
 
   
+
   if (mesh->GetTimeStamp() > surface_vector_timestamp ||
       solutiontimestamp > surface_vector_timestamp ||
       zoomall)
@@ -509,7 +499,11 @@ namespace netgen
       if (vispar.clipenable && clipsolution)      
 	{
 	  // lock->UnLock();
+	  NgLock mlock (mesh->Mutex(), 0);
+	  mlock.UnLock();
 	  mesh->BuildElementSearchTree();
+	  mlock.Lock();
+
 	  // lock->Lock();
 	}
 
@@ -531,7 +525,7 @@ namespace netgen
 	  
 	  glBegin (GL_TRIANGLES);
 	  for (i = 0; i < cpt.Size(); i++)
-	    DrawClipPlaneTrig (sol, scalcomp, cpt[i], 2*subdivisions);
+	    DrawClipPlaneTrig (sol, scalcomp, cpt[i], 0); // 2*subdivisions);
 	  glEnd();
 
 	  glEnable(GL_CLIP_PLANE0);
@@ -544,6 +538,7 @@ namespace netgen
 	    GetMinMax (vecfunction, 0, minval, maxval);
 
 
+	  bool drawelem;
 	  ARRAY<ClipPlanePoint> cpp;
 	  GetClippingPlaneGrid (cpp);
 
@@ -553,12 +548,13 @@ namespace netgen
 	      double values[6];
 	      Vec3d v;
 
-	      GetValues (vsol, p.elnr, p.lam1, p.lam2, p.lam3, values);
+	      drawelem = GetValues (vsol, p.elnr, p.lam1, p.lam2, p.lam3, values);
 	      RealVec3d (values, v, vsol->iscomplex, imag_part);
 
 	      double val = v.Length();
 
-	      if (val > 1e-10 * maxval)
+	      // "drawelem": added 07.04.2004 (FB)
+	      if (drawelem && val > 1e-10 * maxval)
 		{
 		  v *= (rad / val / gridsize * 0.5);
 		  
@@ -571,9 +567,6 @@ namespace netgen
       glEndList ();
     }
 
-  
-  
-
   if (
       numisolines && 
       (clipplanetimestamp < vispar.clipplanetimestamp ||
@@ -588,116 +581,162 @@ namespace netgen
       Point<3> points[1100];
       double values[1100];
       
-      SurfaceElementIndex sei;
-      glBegin (GL_LINES);
       int nse = mesh->GetNSE();
-      if (sol)
-	for (sei = 0; sei <= nse; sei++)
-	  {
-	    const Element2d & el = (*mesh)[sei];
-	    
-	    if (el.GetType() == TRIG || el.GetType() == TRIG6)
-	      {
-		Point<3> lp1, lp2, lp3;
-		if (!mesh->GetCurvedElements().IsHighOrder())
-		  {
-		    GetPointDeformation (el[0]-1, lp1);
-		    GetPointDeformation (el[1]-1, lp2);
-		    GetPointDeformation (el[2]-1, lp3);
-		  }
-
 
-		
-		int n = 1 << subdivisions;
-		int ii = 0;
-		int ix, iy;
-		for (iy = 0; iy <= n; iy++)
-		  for (ix = 0; ix <= n-iy; ix++)
+      if (sol)
+	{
+	  glBegin (GL_LINES);
+	  
+	  for (SurfaceElementIndex sei = 0; sei < nse; sei++)
+	    {
+	      const Element2d & el = (*mesh)[sei];
+	      
+	      if (el.GetType() == TRIG || el.GetType() == TRIG6)
+		{
+		  Point<3> lp1, lp2, lp3;
+		  if (!mesh->GetCurvedElements().IsHighOrder())
 		    {
-		      double x = double(ix) / n;
-		      double y = double(iy) / n;
-		  
-		      values[ii] = GetSurfValue (sol, sei, x, y, scalcomp);
-		      Point<2> xref(x,y);
-		      
-		      if (mesh->GetCurvedElements().IsHighOrder())
-			mesh->GetCurvedElements().
-			  CalcSurfaceTransformation (xref, sei, points[ii]);
-		      else
-			points[ii] = lp3 + x * (lp1-lp3) + y * (lp2-lp3);
-		  
-		      if (deform)
-			{
-			  Vec<3> def;
-			  GetSurfDeformation (sei, x, y, def);
-			  points[ii] += def;
-			}
-		      ii++;
+		      GetPointDeformation (el[0]-1, lp1);
+		      GetPointDeformation (el[1]-1, lp2);
+		      GetPointDeformation (el[2]-1, lp3);
 		    }
-
-		ii = 0;
-		for (iy = 0; iy < n; iy++)
-		  {
-		    for (ix = 0; ix < n-iy; ix++)
+		  
+		  int n = 1 << subdivisions;
+		  int ii = 0;
+		  int ix, iy;
+		  for (iy = 0; iy <= n; iy++)
+		    for (ix = 0; ix <= n-iy; ix++)
 		      {
 			double x = double(ix) / n;
 			double y = double(iy) / n;
 			
+			// TODO: consider return value (bool: draw/don't draw element)
+			GetSurfValue (sol, sei, x, y, scalcomp, values[ii]);
+			Point<2> xref(x,y);
+			
+			if (mesh->GetCurvedElements().IsHighOrder())
+			  mesh->GetCurvedElements().
+			    CalcSurfaceTransformation (xref, sei, points[ii]);
+			else
+			  points[ii] = lp3 + x * (lp1-lp3) + y * (lp2-lp3);
+			
+			if (deform)
+			  {
+			    Vec<3> def;
+			    GetSurfDeformation (sei, x, y, def);
+			    points[ii] += def;
+			  }
+			ii++;
+		      }
+		  
+		  ii = 0;
+		  for (iy = 0; iy < n; iy++, ii++)
+		    for (ix = 0; ix < n-iy; ix++, ii++)
+		      {
 			int index[] = { ii, ii+1, ii+n-iy+1,
 					ii+1, ii+n-iy+2, ii+n-iy+1 };
 			
-			int np = (ix == n-iy-1) ? 3 : 6;
-
 			DrawIsoLines (points[index[0]], points[index[1]], points[index[2]],
 				      values[index[0]], values[index[1]], values[index[2]],
 				      minval, maxval, numisolines);
-			if (np == 6)
+			if (ix < n-iy-1) 
 			  DrawIsoLines (points[index[3]], points[index[4]], points[index[5]],
-					values[index[3]], values[index[4]], values[index[5]],
+				      values[index[3]], values[index[4]], values[index[5]],
 					minval, maxval, numisolines);
-			ii++;
+		      }    
+		}
+	      
+	      
+	      if (el.GetType() == QUAD || el.GetType() == QUAD6 || el.GetType() == QUAD8 )
+		{
+		  Point<3> lpi[4];
+		  Vec<3> vx, vy, vtwist, def;
+		  if (!mesh->GetCurvedElements().IsHighOrder())
+		    {
+		      for (int j = 0; j < 4; j++)
+			GetPointDeformation (el[j]-1, lpi[j]);
+		      vx = lpi[1]-lpi[0];
+		      vy = lpi[3]-lpi[0];
+		      vtwist = (lpi[0]-lpi[1]) + (lpi[2]-lpi[3]);
+		    }
+
+		  int n = 1 << subdivisions;
+		  int ix, iy, ii = 0;
+		  for (iy = 0; iy <= n; iy++)
+		    for (ix = 0; ix <= n; ix++, ii++)
+		      {
+			double x = double(ix) / n;
+			double y = double(iy) / n;
+			
+			// TODO: consider return value (bool: draw/don't draw element)
+			GetSurfValue (sol, sei, x, y, scalcomp, values[ii]);
+			Point<2> xref(x,y);
+			
+			if (mesh->GetCurvedElements().IsHighOrder())
+			  mesh->GetCurvedElements().
+			    CalcSurfaceTransformation (xref, sei, points[ii]);
+			else
+			  points[ii] = lpi[0] + x * vx + y * vy + x*y * vtwist;
+			
+			if (deform)
+			  {
+			    GetSurfDeformation (sei, x, y, def);
+			    points[ii] += def;
+			  }
+		      }
+		  
+		  ii = 0;
+		  for (iy = 0; iy < n; iy++, ii++)
+		    for (ix = 0; ix < n; ix++, ii++)
+		      {
+			DrawIsoLines (points[ii], points[ii+1], points[ii+n+1],
+				      values[ii], values[ii+1], values[ii+n+1],
+				      minval, maxval, numisolines);
+			DrawIsoLines (points[ii+1], points[ii+n+2], points[ii+n+1],
+				      values[ii+1], values[ii+n+2], values[ii+n+1],
+				      minval, maxval, numisolines);
 		      }	
-		    ii++;
-		  }    
-		/*
-		double val1, val2, val3;
-		val1 = GetSurfValue (sol, sei, 1, 0, scalcomp);
-		val2 = GetSurfValue (sol, sei, 0, 1, scalcomp);
-		val3 = GetSurfValue (sol, sei, 0, 0, scalcomp);
-		*/
-	      }
-	  }
-      
+		}
+	    }
+	  glEnd();
+	}
+      glEndList ();
+
+      if (clipplane_isolinelist) glDeleteLists (clipplane_isolinelist, 1);
+            
       if (vispar.clipenable && clipsolution == 1 && sol)
 	{
-	  glDisable(GL_CLIP_PLANE0);
-	  
+	  clipplane_isolinelist = glGenLists (1);
+	  glNewList (clipplane_isolinelist, GL_COMPILE);
+
 	  ARRAY<ClipPlaneTrig> cpt;
 	  GetClippingPlaneTrigs (cpt);
+	  bool drawelem;
 	  
 	  glNormal3d (-clipplane[0], -clipplane[1], -clipplane[2]);
 	  
 	  if (numisolines)
-	  for (i = 0; i < cpt.Size(); i++)
-	    {
-	      const ClipPlaneTrig & trig = cpt[i];
-	      double vali[3];
-	      for (j = 0; j < 3; j++)
-		vali[j] = GetValue (sol, trig.elnr, 
-				    trig.points[j].lami(0),
-				    trig.points[j].lami(1),
-				    trig.points[j].lami(2), scalcomp);
-	      
-	      DrawIsoLines (trig.points[0].p,
-			    trig.points[1].p,
-			    trig.points[2].p,
-			    vali[0], vali[1], vali[2], minval, maxval, numisolines);
-	    }
-	  glEnable(GL_CLIP_PLANE0);
+	    for (i = 0; i < cpt.Size(); i++)
+	      {
+		const ClipPlaneTrig & trig = cpt[i];
+		double vali[3];
+		for (j = 0; j < 3; j++)
+		  drawelem = GetValue (sol, trig.elnr, 
+				       trig.points[j].lami(0),
+				       trig.points[j].lami(1),
+				     trig.points[j].lami(2), scalcomp, vali[j]);
+		
+		if ( drawelem )
+		  DrawIsoLines (trig.points[0].p,
+				trig.points[1].p,
+				trig.points[2].p,
+				vali[0], vali[1], vali[2], minval, maxval, numisolines);
+	      }
+	  glEndList ();
 	}
       glEnd();
       
-      glEndList ();
+
     }
   
   clipplanetimestamp = max2 (vispar.clipplanetimestamp, solutiontimestamp);
@@ -726,7 +765,7 @@ namespace netgen
     mesh->GetBox (pmin, pmax);
     double lami[3];
     int i;
-
+    bool drawelem;
 
     fieldlineslist = glGenLists (num_fieldlineslists);
 
@@ -750,7 +789,7 @@ namespace netgen
 	      {
 		Vec3d v;
 		double values[6];
-		GetValues (vsol, elnr, lami[0], lami[1], lami[2], values);
+		drawelem = GetValues (vsol, elnr, lami[0], lami[1], lami[2], values);
 		RealVec3d (values, v, vsol->iscomplex, phaser, phasei);
 		
 		double val = v.Length();
@@ -765,7 +804,8 @@ namespace netgen
 		    
 		    Point3d p2 = p + v; 
 		    cout << " p " << p << endl; 
-		    DrawCylinder (p, p2, rad * 0.003);
+		    // "drawelem": added 07.04.2004 (FB)
+		    if ( drawelem ) DrawCylinder (p, p2, rad * 0.003);
 		    p = p2;  		
 		    
 		    for(i=0;i<20;i++) 
@@ -774,14 +814,15 @@ namespace netgen
 			
 			if (elnr != -1)
 			  {
-			    GetValues (vsol, elnr, lami[0], lami[1], lami[2], values);
+			    drawelem = GetValues (vsol, elnr, lami[0], lami[1], lami[2], values);
 			    RealVec3d (values, v, vsol->iscomplex, phaser, phasei);
 			    val = v.Length();
 			    v *= (rad / val * 0.02);
 			    
 			    SetOpenGlColor  (val, minval, maxval, logscale);
 			    p2 = p +v; 
-			    DrawCylinder (p, p2, rad * 0.003);
+			    // "drawelem": added 07.04.2004 (FB)
+			    if ( drawelem ) DrawCylinder (p, p2, rad * 0.003);
 			    p = p2;  	
 			  }
 			else break; 
@@ -793,7 +834,7 @@ namespace netgen
 
 			if (elnr != -1)
 			  {
-			    GetValues (vsol, elnr, lami[0], lami[1], lami[2], values);
+			    drawelem = GetValues (vsol, elnr, lami[0], lami[1], lami[2], values);
 			    RealVec3d (values, v, vsol->iscomplex, phaser, phasei);
 			    
 			    val = v.Length();
@@ -801,7 +842,8 @@ namespace netgen
 			    
 			    SetOpenGlColor  (val, minval, maxval, logscale);
 			    p2 = p - v; 
-			    DrawCylinder (p, p2, rad * 0.003);
+			    // "drawelem": added 07.04.2004 (FB)
+			    if ( drawelem ) DrawCylinder (p, p2, rad * 0.003);
 			    p = p2;  	
 			  }
 			else break; 
@@ -815,13 +857,12 @@ namespace netgen
   
 
 
+
   void  VisualSceneSolution :: DrawSurfaceElements ()
   {
-    int j, k;
-    SurfaceElementIndex sei;
-
     const SolData * sol = NULL;
     const SolData * vsol = NULL;
+    bool drawelem = 0;
     
     if (scalfunction != -1)
       sol = soldata[scalfunction];
@@ -833,27 +874,36 @@ namespace netgen
 	sol = NULL;
 	vsol = NULL;
       }
-    
+
     glLineWidth (1.0f);
 
     if (!sol || !sol->draw_surface)
       glDisable (GL_TEXTURE_1D);
-
-    
+        
     Point<3> points[1100];
     Vec<3> nvs[1100];
     double values[1100];
 
-    glBegin (GL_TRIANGLES);
     int nse = mesh->GetNSE();
 
-    glColor3f (0.4, 0.4, 0.4);
+    glBegin (GL_TRIANGLES);
 
-    for (sei = 0; sei < nse; sei++)
+    // glColor3f (0.4, 0.4, 0.4);
+    // glColor3d (0.8, 0.8, 0.8);
+    glColor3d (1.0, 1.0, 1.0);
+
+    for(SurfaceElementIndex sei = 0; sei < nse; sei++)
       {
-	Element2d & el = (*mesh)[sei];
+	const Element2d & el = (*mesh)[sei];
 
-	if (el.GetType() == TRIG || el.GetType() == TRIG6)
+	if(vispar.drawdomainsurf > 0 &&
+	   ((mesh->GetDimension() == 3 && 
+	     vispar.drawdomainsurf != mesh->GetFaceDescriptor(el.GetIndex()).DomainIn() &&
+	     vispar.drawdomainsurf != mesh->GetFaceDescriptor(el.GetIndex()).DomainOut()) ||
+	    (mesh->GetDimension() == 2 && el.GetIndex() != vispar.drawdomainsurf))) continue;
+	
+	
+	if ( el.GetType() == TRIG || el.GetType() == TRIG6 )
 	  {
 	    Point<3> p1, p2, p3;
 	    if (!mesh->GetCurvedElements().IsHighOrder())
@@ -865,15 +915,14 @@ namespace netgen
 	    
 	    int n = 1 << subdivisions;
 	    int ii = 0;
-	    int ix, iy;
-	    for (iy = 0; iy <= n; iy++)
-	      for (ix = 0; ix <= n-iy; ix++)
+	    for (int iy = 0; iy <= n; iy++)
+	      for (int ix = 0; ix <= n-iy; ix++)
 		{
 		  double x = double(ix) / n;
 		  double y = double(iy) / n;
 		  
 		  if (sol && sol->draw_surface) 
-		    values[ii] = GetSurfValue (sol, sei, x, y, scalcomp);
+		    drawelem = GetSurfValue (sol, sei, x, y, scalcomp, values[ii]);
 
 		  Point<2> xref(x,y);
 		  Mat<3,2> dxdxi;
@@ -902,125 +951,129 @@ namespace netgen
 		}
 
 	    ii = 0;
-	    
-	    for (iy = 0; iy < n; iy++)
-	      {
-		for (ix = 0; ix < n-iy; ix++)
-		  {
-		    double x = double(ix) / n;
-		    double y = double(iy) / n;
-		    
-		    int index[] = { ii, ii+1, ii+n-iy+1,
-				    ii+1, ii+n-iy+2, ii+n-iy+1 };
-		    
-		    int np = (ix == n-iy-1) ? 3 : 6;
-		    for (int j = 0; j < np; j++)
-		      {
-			if (sol && sol->draw_surface)
-			  SetOpenGlColor  (values[index[j]], minval, maxval, logscale);
-			glNormal3dv (&nvs[index[j]](0));
-			glVertex3dv (points[index[j]]);
-		      }
-		    ii++;
-		  }	
-		ii++;
-	      }    
+	    for (int iy = 0; iy < n; iy++, ii++)
+	      for (int ix = 0; ix < n-iy; ix++, ii++)
+		{
+		  double x = double(ix) / n;
+		  double y = double(iy) / n;
+		  
+		  int index[] = { ii, ii+1, ii+n-iy+1,
+				  ii+1, ii+n-iy+2, ii+n-iy+1 };
+		  
+		  int np = (ix == n-iy-1) ? 3 : 6;
+		  for (int j = 0; j < np; j++)
+		    {
+		      if (sol && sol->draw_surface && drawelem)
+			SetOpenGlColor  (values[index[j]], minval, maxval, logscale);
+		      else
+			glColor3f (0.4f, 0.4f, 0.4f);
+		      
+		      glNormal3dv (nvs[index[j]]);
+		      glVertex3dv (points[index[j]]);
+		    }
+		}	
 	  }
       }
     glEnd ();
     
-    
-  
+      
     
   
     
     glBegin (GL_QUADS);
-    for (sei = 0; sei < nse; sei++)
-    {
-      const Element2d & el = (*mesh)[sei];
-      
-      if (el.GetType() == QUAD || el.GetType() == QUAD6)
-	{
-	  Point<3> lpi[4];
-	  for (k = 0; k < 4; k++)
-	    GetPointDeformation (el[k]-1, lpi[k]);
-	  Vec<3> vx = lpi[1]-lpi[0];
-	  Vec<3> vy = lpi[3]-lpi[0];
-	  Vec<3> vtwist = (lpi[0]-lpi[1]) + (lpi[2]-lpi[3]);
-	  Vec<3> nv = Cross (lpi[1]-lpi[0], Center (lpi[2],lpi[3]) - lpi[0]);
-	  nv.Normalize();
-	  glNormal3dv (nv);
-	  
+    for (SurfaceElementIndex sei = 0; sei < nse; sei++)
+      {
+	const Element2d & el = (*mesh)[sei];
 
-	  int n = 1 << subdivisions;
-	  int ii = 0;
-	  int ix, iy;
-	  for (iy = 0; iy <= n; iy++)
-	    for (ix = 0; ix <= n; ix++)
+	if(vispar.drawdomainsurf > 0 &&
+	   ((mesh->GetDimension() == 3 && 
+	     vispar.drawdomainsurf != mesh->GetFaceDescriptor(el.GetIndex()).DomainIn() &&
+	     vispar.drawdomainsurf != mesh->GetFaceDescriptor(el.GetIndex()).DomainOut()) ||
+	    (mesh->GetDimension() == 2 && el.GetIndex() != vispar.drawdomainsurf))) continue;
+
+	if ( el.GetType() == QUAD || el.GetType() == QUAD6 )
+	  {
+	    Point<3> lpi[4];
+	    Vec<3> vx, vy, vtwist;
+	    
+	    if (!mesh->GetCurvedElements().IsHighOrder())
 	      {
-		double x = double(ix) / n;
-		double y = double(iy) / n;
+		for (int k = 0; k < 4; k++)
+		  GetPointDeformation (el[k]-1, lpi[k]);
 		
-		Point<2> xref(x,y);
-		Mat<3,2> dxdxi;
+		vx = lpi[1]-lpi[0];
+		vy = lpi[3]-lpi[0];
+		vtwist = (lpi[0]-lpi[1]) + (lpi[2]-lpi[3]);
+	      }
+	    Vec<3> nv = Cross (lpi[1]-lpi[0], Center (lpi[2],lpi[3]) - lpi[0]);
+	    nv.Normalize();
+	    glNormal3dv (nv);
 
-		if (sol && sol->draw_surface) 
-		  values[ii] = GetSurfValue (sol, sei, x, y, scalcomp);
+	    int n = 1 << subdivisions;
+	    int ii = 0;
+	    int ix, iy;
+	    for (iy = 0; iy <= n; iy++)
+	      for (ix = 0; ix <= n; ix++)
+		{
+		  double x = double(ix) / n;
+		  double y = double(iy) / n;
+		
+		  Point<2> xref(x,y);
+		  Mat<3,2> dxdxi;
 
+		  if (sol && sol->draw_surface) 
+		    drawelem = GetSurfValue (sol, sei, x, y, scalcomp, values[ii]);
 
-		if (mesh->GetCurvedElements().IsHighOrder())
-		  {
-		    mesh->GetCurvedElements().
-		      CalcSurfaceTransformation (xref, sei, points[ii], dxdxi);
-		    nvs[ii] = Cross (dxdxi.Col(0), dxdxi.Col(1));
-		    nvs[ii].Normalize();
-		  }
-		else
-		  {
-		    points[ii] = lpi[0] + x * vx + y * vy + x*y * vtwist;
-		    nvs[ii] = Cross (vx, vy);
-		    nvs[ii].Normalize();
-		  }
+		  if (mesh->GetCurvedElements().IsHighOrder())
+		    {
+		      mesh->GetCurvedElements().
+			CalcSurfaceTransformation (xref, sei, points[ii], dxdxi);
+		      nvs[ii] = Cross (dxdxi.Col(0), dxdxi.Col(1));
+		      nvs[ii].Normalize();
+		    }
+		  else
+		    {
+		      points[ii] = lpi[0] + x * vx + y * vy + x*y * vtwist;
+		      nvs[ii] = Cross (vx, vy);
+		      nvs[ii].Normalize();
+		    }
 		  
-		if (deform)
-		  {
-		    Vec<3> def;
-		    GetSurfDeformation (sei, x, y, def);
-		    points[ii] += def;
-		  }
+		  if (deform)
+		    {
+		      Vec<3> def;
+		      GetSurfDeformation (sei, x, y, def);
+		      points[ii] += def;
+		    }
 
-		// points[ii] = lpi[0] + x * vx + y * vy;
-		ii++;
-	      }
+		  ii++;
+		}
 	  
-	  ii = 0;
-	  for (iy = 0; iy < n; iy++)
-	    {
-	      for (ix = 0; ix < n; ix++)
+	    ii = 0;
+	    for (iy = 0; iy < n; iy++, ii++)
+	      for (ix = 0; ix < n; ix++, ii++)
 		{
 		  double x = double(ix) / n;
 		  double y = double(iy) / n;
-		    
+		  
 		  int index[] = { ii, ii+1, ii+n+2, ii+n+1 };
-
+		  
 		  for (int j = 0; j < 4; j++)
 		    {
-		      if (sol && sol->draw_surface)
+		      if (sol && sol->draw_surface && drawelem)
 			SetOpenGlColor  (values[index[j]], minval, maxval, logscale);
-		      glNormal3dv (&nvs[index[j]](0));
+		      else
+			glColor3f (0.4f, 0.4f, 0.4f);
+		      
+		      glNormal3dv (nvs[index[j]]);
 		      glVertex3dv (points[index[j]]);
 		    }
-		  ii++;
 		}	
-	      ii++;
-	    }    
-	}
-    }
-  glEnd();
-
-  if (usetexture)
-    glEnable (GL_TEXTURE_1D);
-
+	  }
+      }
+    glEnd();
+    
+    if (usetexture)
+      glEnable (GL_TEXTURE_1D);
   }
 
 
@@ -1069,21 +1122,21 @@ nverse (mat, invmat);
 	  nv = 4;
 	
 	Point<3> p1, p2, p3, p4;
-	GetPointDeformation (el[0]-1, p1, sei);
-	GetPointDeformation (el[1]-1, p2, sei);
-	GetPointDeformation (el[2]-1, p3, sei);
-	if (nv == 4)
-	  GetPointDeformation (el[3]-1, p4, sei);
+	if (!mesh->GetCurvedElements().IsHighOrder())
+	  {
+	    p1 = (*mesh)[el[0]];
+	    p2 = (*mesh)[el[1]];
+	    p3 = (*mesh)[el[2]];
+	    if (nv == 4)
+	      p4 = (*mesh)[el[3]];
+	  }
 
 
 	// glBegin (GL_LINE_LOOP);
 	int n = 1 << subdivisions;
 	// n = p;
 
-	int ii = 0;
-	int ix, iy;
 	Point<3> pnt;
-	
 	for (k = 0; k < nv; k++)
 	  {
 	    Point<2> p0;
@@ -1130,16 +1183,15 @@ nverse (mat, invmat);
 	    Point<3> pts[33], pts2[33];
 	    if (n > 32) cerr << "too many subdivisions, code 433425" << endl;
 
-	    for (ix = 0; ix <= n; ix++)
+	    for (int ix = 0; ix <= n; ix++)
 	      {
 		Point<2> p = p0 + (double(ix) / n) * vtau;
 		double x = p(0);
 		double y = p(1);
-		Point<2> xref(x,y);
 
 		if (mesh->GetCurvedElements().IsHighOrder())
 		  mesh->GetCurvedElements().
-		    CalcSurfaceTransformation (xref, sei, pnt);
+		    CalcSurfaceTransformation (p, sei, pnt);
 		else
 		  {
 		    if (nv == 3)
@@ -1163,6 +1215,8 @@ nverse (mat, invmat);
 
 
 
+
+
 	    /*
 	    // convert from point-values to Bernstein basis
 	    for (i = 0; i < 3; i++)
@@ -1220,6 +1274,7 @@ void  VisualSceneSolution :: DrawSurfaceVectors ()
   SurfaceElementIndex sei;
 
   const SolData * vsol = NULL;
+  bool drawelem;
 
   if (vecfunction != -1)
     vsol = soldata[vecfunction];
@@ -1244,6 +1299,7 @@ void  VisualSceneSolution :: DrawSurfaceVectors ()
       vsol->soltype==SOL_SURFACE_NONCONTINUOUS ||
       vsol->soltype==SOL_VIRTUALFUNCTION)
   */
+
   if (vsol->draw_surface && showsurfacesolution)
     {
       int nse = mesh->GetNSE();
@@ -1304,11 +1360,12 @@ void  VisualSceneSolution :: DrawSurfaceVectors ()
 	  double inv12 = -mat12/det;
 	  double inv22 = mat11/det;
 	  
-      
+	  //	  cout << "drawsurfacevectors. xoffset = " << xoffset << ", yoffset = ";
+	  //	  cout << yoffset << endl;
 	  
-	  for (s = 0; s <= 1; s += 1.0 / gridsize)
+	  for (s = xoffset/gridsize; s <= 1+xoffset/gridsize; s += 1.0 / gridsize)
 	    if (s >= minx2d && s <= maxx2d)
-	      for (t = 0; t <= 1; t += 1.0 / gridsize)
+	      for (t = yoffset/gridsize; t <= 1+yoffset/gridsize; t += 1.0 / gridsize)
 		if (t >= miny2d && t <= maxy2d)
 		  {
 		    double lam1 = inv11 * (s - p2d[0](0)) + inv12 * (t-p2d[0](1));
@@ -1324,7 +1381,7 @@ void  VisualSceneSolution :: DrawSurfaceVectors ()
 
 			Vec<3> v;
 			double values[6];
-			GetSurfValues (vsol, sei, lam1, lam2, values);
+			drawelem = GetSurfValues (vsol, sei, lam1, lam2, values);
 
 			if (!vsol->iscomplex)
 			  for (k = 0; k < 3; k++)
@@ -1340,14 +1397,18 @@ void  VisualSceneSolution :: DrawSurfaceVectors ()
 			  }
 			
 			if (mesh->GetDimension() == 2)
-			  v(2) = 0;
+			  if ( (!vsol->iscomplex && vsol->components != 3) ||
+			       (vsol->iscomplex && vsol->components != 6) )
+			    v(2) = 0;
 			
 			double val = v.Length();
 			SetOpenGlColor  (val, minval, maxval, logscale);
 
 			if (val > 1e-10 * maxval)
 			  v *= (rad / val / gridsize * 0.5);
-			DrawCone (cp, cp+4*v, 0.8*rad / gridsize);
+			// "drawelem": added 07.04.2004 (FB)
+			if ( drawelem ) DrawCone (cp, cp+4*v, 0.8*rad / gridsize);
+
 
 			/*
 			v /= val;
@@ -1440,8 +1501,9 @@ GetMinMax (int funcnr, int comp, double & minv, double & maxv) const
   int i, j;
   const SolData * sol;
   double val;
+  bool considerElem;
 
-  bool hasit = 0;
+  bool hasit = false;
   minv = 0; maxv = 1;
   if (funcnr != -1)
     {
@@ -1451,12 +1513,16 @@ GetMinMax (int funcnr, int comp, double & minv, double & maxv) const
 	  int ne = mesh->GetNE();
 	  for (int i = 0; i < ne; i++)
 	    {
-	      double val = GetValue (sol, i, 0.333, 0.333, 0.333, comp);
-	      if (val > maxv || !hasit)
-		maxv = val;
-	      if (val < minv || !hasit)
-		minv = val;
-	      hasit = 1;
+	      // "considerElem": added 07.04.2004 (FB)
+	      considerElem = GetValue (sol, i, 0.333, 0.333, 0.333, comp, val);
+	      if (considerElem)
+		{
+		  if (val > maxv || !hasit)
+		    maxv = val;
+		  if (val < minv || !hasit)
+		    minv = val;
+		  hasit = true;
+		}
 	    }
 	}
       if (sol->draw_surface)
@@ -1464,12 +1530,16 @@ GetMinMax (int funcnr, int comp, double & minv, double & maxv) const
 	  int nse = mesh->GetNSE();
 	  for (int i = 0; i < nse; i++)
 	    {
-	      double val = GetSurfValue (sol, i, 0.333, 0.333, comp);
-	      if (val > maxv || !hasit)
-		maxv = val;
-	      if (val < minv || !hasit)
-		minv = val;
-	      hasit = 1;
+	      // "considerElem": added 07.04.2004 (FB)
+	      considerElem = GetSurfValue (sol, i, 0.333, 0.333, comp, val);
+	      if (considerElem)
+		{
+		  if (val > maxv || !hasit)
+		    maxv = val;
+		  if (val < minv || !hasit)
+		    minv = val;
+		  hasit = true;
+		}
 	    }
 	}
     }
@@ -1479,38 +1549,41 @@ GetMinMax (int funcnr, int comp, double & minv, double & maxv) const
 
 
 
-void VisualSceneSolution :: 
+bool VisualSceneSolution :: 
 GetValues (const SolData * data, ElementIndex elnr, 
 	   double lam1, double lam2, double lam3,
 	   double * values) const
 {
+  bool ok;
   switch (data->soltype)
     {
     case SOL_VIRTUALFUNCTION:
       {
-	bool ok =
-	  data->solclass->GetValue (elnr, lam1, lam2, lam3, values);
+	ok = data->solclass->GetValue (elnr, lam1, lam2, lam3, values);
 	break;
       }
     default:
       {
 	for (int i = 0; i < data->components; i++)
-	  values[i] = GetValue (data, elnr, lam1, lam2, lam3, i+1);
+	  ok = GetValue (data, elnr, lam1, lam2, lam3, i+1, values[i]);
       }
     }
+  return ok;
 }
 
 
-double VisualSceneSolution :: 
+bool VisualSceneSolution :: 
 GetValue (const SolData * data, ElementIndex elnr, 
 	  double lam1, double lam2, double lam3,
-	  int comp) const
+	  int comp, double & val) const
 {
+  val = 0;
+  bool ok = 0;
+
   if (comp == 0)
     {
-      double val = 0;
       ArrayMem<double,20> values(data->components);
-      GetValues (data, elnr, lam1, lam2, lam3, &values[0]);
+      ok = GetValues (data, elnr, lam1, lam2, lam3, &values[0]);
 
       switch (evalfunc)
 	{
@@ -1596,7 +1669,7 @@ GetValue (const SolData * data, ElementIndex elnr,
          }
       }
 
-      return val;	
+      return ok;	
     }
 
 
@@ -1605,12 +1678,10 @@ GetValue (const SolData * data, ElementIndex elnr,
     case SOL_VIRTUALFUNCTION:
       {
 	double values[20];
-	bool ok = data->solclass->GetValue (elnr, lam1, lam2, lam3, values);
+	ok = data->solclass->GetValue (elnr, lam1, lam2, lam3, values);
 
-	if (ok) 
-	  return values[comp-1];
-	else
-	  return 0;
+	val = values[comp-1];
+	return ok;
       }
     case SOL_NODAL:
       {
@@ -1618,7 +1689,6 @@ GetValue (const SolData * data, ElementIndex elnr,
 
 	double lami[8];
 	int np, i;
-	double val = 0;
 	
 	switch (el.GetType())
 	  {
@@ -1649,12 +1719,13 @@ GetValue (const SolData * data, ElementIndex elnr,
 	for (i = 0; i < np; i++)
 	  val += lami[i] * data->data[(el[i]-1) * data->dist + comp-1];
 
-	return val;
+	return 1;
       }
 
     case SOL_ELEMENT:
       {
-	return data->data[elnr * data->dist + comp-1];
+	val = data->data[elnr * data->dist + comp-1];
+	return 1;
       }
 
     case SOL_SURFACE_ELEMENT:
@@ -1666,7 +1737,6 @@ GetValue (const SolData * data, ElementIndex elnr,
 
 	double lami[8];
 	int np, i;
-	double val = 0;
 
 	switch (el.GetType())
 	  {
@@ -1726,14 +1796,20 @@ GetValue (const SolData * data, ElementIndex elnr,
 	for (i = 0; i < np; i++)
 	  val += lami[i] * data->data[(base+i) * data->dist + comp-1];
 
-	return val;
+	return 1;
       }
 
     case SOL_MARKED_ELEMENTS:
-      return (*mesh)[elnr].TestRefinementFlag();
+      {
+	val = (*mesh)[elnr].TestRefinementFlag();
+	return 1;
+      }
       
     case SOL_ELEMENT_ORDER:
-      return (*mesh)[elnr].GetOrder();
+      {
+	val = (*mesh)[elnr].GetOrder();
+	return 1;
+      }
     }
   return 0;
 }
@@ -1741,39 +1817,41 @@ GetValue (const SolData * data, ElementIndex elnr,
 
 
 
-void VisualSceneSolution :: 
+bool VisualSceneSolution :: 
 GetSurfValues (const SolData * data, SurfaceElementIndex selnr, 
 	       double lam1, double lam2, 
 	       double * values) const
 {
+  bool ok;
   switch (data->soltype)
     {
     case SOL_VIRTUALFUNCTION:
       {
-	bool ok =
-	  data->solclass->GetSurfValue (selnr, lam1, lam2, values);
+	ok = data->solclass->GetSurfValue (selnr, lam1, lam2, values);
 	break;
       }
     default:
       {
 	for (int i = 0; i < data->components; i++)
-	  values[i] = GetSurfValue (data, selnr, lam1, lam2, i+1);
+	  ok = GetSurfValue (data, selnr, lam1, lam2, i+1, values[i]);
       }
     }
+  return ok;
 }
 
 
 
-double VisualSceneSolution :: 
+bool VisualSceneSolution :: 
 GetSurfValue (const SolData * data, SurfaceElementIndex selnr, 
 	      double lam1, double lam2, 
-	      int comp) const
+	      int comp, double & val) const
 {
+  bool ok;
   if (comp == 0)
     {
-      double val = 0;
+      val = 0;
       ArrayMem<double,20> values(data->components);
-      GetSurfValues (data, selnr, lam1, lam2, &values[0]);
+      ok = GetSurfValues (data, selnr, lam1, lam2, &values[0]);
 
       switch (evalfunc)
 	{
@@ -1859,7 +1937,7 @@ GetSurfValue (const SolData * data, SurfaceElementIndex selnr,
          }
       }
 
-      return val;	
+      return ok;	
 
 
       /*
@@ -1876,13 +1954,23 @@ GetSurfValue (const SolData * data, SurfaceElementIndex selnr,
     {
    case SOL_VIRTUALFUNCTION:
       {
-	double values[20];
-	bool ok =
-	  data->solclass->GetSurfValue (selnr, lam1, lam2, values);
+	ArrayMem<double,20> values(data->components);
+	bool ok;
+
+	ok = data->solclass->GetSurfValue (selnr, lam1, lam2, &values[0]);
+
 	if (ok)
-	  return values[comp-1];
-	else 
-	  return 0;
+	  {
+	    if (!data->iscomplex)
+	      val =  values[comp-1];
+	    else
+	      {
+		// cout << "time = " << time << ", cos = " << cos(time) << endl;
+		val = values[comp-1]*cos(3*time) + values[comp]*sin(3*time);
+	      }
+	  }
+
+	return ok;
       }
 
 
@@ -1892,7 +1980,7 @@ GetSurfValue (const SolData * data, SurfaceElementIndex selnr,
 
 	double lami[8];
 	int np, i;
-	double val = 0;
+	val = 0;
 	double lam3 = 1-lam1-lam2;
 
 	switch (el.GetType())
@@ -1941,7 +2029,7 @@ GetSurfValue (const SolData * data, SurfaceElementIndex selnr,
 	for (i = 0; i < np; i++)
 	  val += lami[i] * data->data[(el[i]-1) * data->dist + comp-1];
 
-	return val;
+	return 1;
       }
 
     case SOL_ELEMENT:
@@ -1950,17 +2038,21 @@ GetSurfValue (const SolData * data, SurfaceElementIndex selnr,
 	mesh->GetTopology().GetSurface2VolumeElement (selnr+1, el1, el2);
 	el1--;
 
-	return data->data[el1 * data->dist+comp-1];
+	val = data->data[el1 * data->dist+comp-1];
+	return 1;
       }
 
     case SOL_NONCONTINUOUS:
       {
+	val = 0;
+	// ?????
 	return 0;
       }
 
     case SOL_SURFACE_ELEMENT:
       {
-	return data->data[selnr * data->dist + comp-1];
+	val = data->data[selnr * data->dist + comp-1];
+	return 1;
       }
 
     case SOL_SURFACE_NONCONTINUOUS:
@@ -1969,7 +2061,7 @@ GetSurfValue (const SolData * data, SurfaceElementIndex selnr,
 
 	double lami[8];
 	int np, i;
-	double val = 0;
+	val = 0;
 	int order = data->order;
 
 	switch (order)
@@ -2031,14 +2123,20 @@ GetSurfValue (const SolData * data, SurfaceElementIndex selnr,
 	  {
 	    val += lami[i] * data->data[(base+i) * data->dist + comp-1];
 	  }
-	return val;
+	return 1;
       }
 
     case SOL_MARKED_ELEMENTS:
-      return (*mesh)[selnr].TestRefinementFlag();
+      {
+	val = (*mesh)[selnr].TestRefinementFlag();
+	return 1;
+      }
       
     case SOL_ELEMENT_ORDER:
-      return (*mesh)[selnr].GetOrder();
+      {
+	val = (*mesh)[selnr].GetOrder();
+	return 1;
+      }
 
     }
   return 0;
@@ -2125,7 +2223,7 @@ void VisualSceneSolution :: GetClippingPlaneTrigs (ARRAY<ClipPlaneTrig> & trigs)
 
   int ii, j, k, l;
   ElementIndex ei;
-  // int np = mesh->GetNV();
+
   int np = mesh->GetNP();
   int ne = mesh->GetNE();
 
@@ -2162,8 +2260,151 @@ void VisualSceneSolution :: GetClippingPlaneTrigs (ARRAY<ClipPlaneTrig> & trigs)
   ARRAY<Element> loctetsloc;
   ARRAY<Point3d> pointsloc;
 
+
+  int n = 1 << subdivisions;
+  ARRAY<Point<3> > grid((n+1)*(n+1)*(n+1));
+  ARRAY<Point<3> > locgrid((n+1)*(n+1)*(n+1));
+  ARRAY<double> val((n+1)*(n+1)*(n+1));
+
+
   for (ei = 0; ei < ne; ei++)
     {
+      ELEMENT_TYPE type = (*mesh)[ei].GetType();
+      if (type == HEX || type == PRISM || type == TET || type == PYRAMID)
+	{
+	  const Element & el = (*mesh)[ei];
+
+	  Vector shape(el.GetNP());
+
+	  int ii = 0;
+	  for (int ix = 0; ix <= n; ix++)
+	    for (int iy = 0; iy <= n; iy++)
+	      for (int iz = 0; iz <= n; iz++, ii++)
+		{
+		  Point<3> ploc;
+		  if (type == PRISM)
+		    ploc = Point<3> (double(ix) / n * (1-double(iy)/n), double(iy) / n, double(iz) / n);
+		  if (type == TET)
+		    ploc = Point<3> (double(ix) / n * (1-double(iy)/n) * (1-double(iz)/n), 
+				     double(iy) / n * (1-double(iz)/n), 
+				     double(iz) / n);
+		  if (type == HEX)
+		    ploc = Point<3> (double(ix) / n, double(iy) / n, double(iz) / n);
+		  if (type == PYRAMID)
+		    ploc = Point<3> (double(ix) / n * (1-double(iz)/n),
+				     double(iy) / n * (1-double(iz)/n),
+				     double(iz)/n);
+
+		  Point<3> pglob;
+		  
+		  if (mesh->GetCurvedElements().IsHighOrder())
+		    {
+		      mesh->GetCurvedElements().
+			CalcElementTransformation (ploc, ei, pglob);
+		    }
+		  else
+		    {
+		      el.GetShapeNew (ploc, shape);
+		      for (int j = 0; j < 3; j++)
+			{
+			  pglob(j) = 0;
+			  for (int k = 0; k < el.GetNP(); k++)
+			    pglob(j) += shape(k) * (*mesh)[el[k]].X(j+1);
+			}
+		    }
+		  
+		  locgrid[ii] = ploc;
+		  grid[ii] = pglob;
+		  val[ii] = 
+		    pglob(0) * clipplane[0] + 
+		    pglob(1) * clipplane[1] + 
+		    pglob(2) * clipplane[2] + 
+		    clipplane[3];
+		}
+	  
+	  for (int ix = 0; ix < n; ix++)
+	    for (int iy = 0; iy < n; iy++)
+	      for (int iz = 0; iz < n; iz++)
+		{
+		  int base = iz + (n+1)*iy + (n+1)*(n+1)*ix;
+		  int pi[8] = 
+		    { base, base+(n+1)*(n+1), base+(n+1)*(n+1)+(n+1), base+(n+1),
+		      base+1, base+(n+1)*(n+1)+1, base+(n+1)*(n+1)+(n+1)+1, base+(n+1)+1 };
+		  
+		  int tets[6][4] = 
+		    { { 1, 7, 2, 3 },
+		      { 1, 7, 3, 4 },
+		      { 1, 7, 4, 8 },
+		      { 1, 7, 8, 5 },
+		      { 1, 7, 5, 6 },
+		      { 1, 7, 6, 2 }
+		    };
+		  
+
+		  for (int ii = 0; ii < 6; ii++)
+		    {
+		      int teti[4];
+		      for (int k = 0; k < 4; k++)
+			teti[k] = pi[tets[ii][k]-1];
+
+		      for (j = 0; j < 4; j++)
+			nodevali[j] = val[teti[j]];
+	  
+		      cntce = 0;
+		      for (j = 0; j < 6; j++)
+			{
+			  int lpi1 = edgei[j][0];
+			  int lpi2 = edgei[j][1];
+			  if ( (nodevali[lpi1] > 0) !=
+			       (nodevali[lpi2] > 0) )
+			    {
+			      edgelam[j] = nodevali[lpi2] / (nodevali[lpi2] - nodevali[lpi1]);
+			      Point<3> p1 = grid[teti[lpi1]];
+			      Point<3> p2 = grid[teti[lpi2]];
+		  
+			      edgep[j] = p1 + (1-edgelam[j]) * (p2-p1);
+		  
+			      cntce++;
+			      cpe3 = cpe2;
+			      cpe2 = cpe1;
+			      cpe1 = j;
+			      if (cntce >= 3)
+				{
+				  ClipPlaneTrig cpt;
+				  cpt.elnr = ei;
+				  
+				  for (int k = 0; k < 3; k++)
+				    {
+				      int ednr;
+				      switch (k)
+					{
+					case 0: ednr = cpe1; break;
+					case 1: ednr = cpe2; break;
+					case 2: ednr = cpe3; break;
+					}
+				      cpt.points[k].p = edgep[ednr];
+				      
+				      int pi1 = edgei[ednr][0];
+				      int pi2 = edgei[ednr][1];
+				      Point<3> p1 = locgrid[teti[pi1]];
+				      Point<3> p2 = locgrid[teti[pi2]];
+				      for (l = 0; l < 3; l++)
+					cpt.points[k].lami(l) = 
+					  edgelam[ednr]     * p1(l) + 
+					  (1-edgelam[ednr]) * p2(l);
+				    }
+				  
+				  trigs.Append (cpt);
+				}
+			    }
+			}
+		    }
+		}
+	}
+
+      else
+	{
+
       // const Element & el = mesh->VolumeElement(i);
 
       (*mesh)[ei].GetTets (loctets);
@@ -2228,6 +2469,7 @@ void VisualSceneSolution :: GetClippingPlaneTrigs (ARRAY<ClipPlaneTrig> & trigs)
 		}
 	    }
 	}
+	}
       
     }
 }
@@ -2258,8 +2500,13 @@ void VisualSceneSolution :: GetClippingPlaneGrid (ARRAY<ClipPlanePoint> & pts)
   int elnr;
   double lami[3];
 
-  for (xi1 = xi1mid-rad; xi1 <= xi1mid+rad; xi1 += rad / gridsize)
-    for (xi2 = xi2mid-rad; xi2 <= xi2mid+rad; xi2 += rad / gridsize)
+  //  cout << "getclippingplanegrid. xoffset = " << xoffset << ", yoffset = ";
+  //  cout << yoffset << endl;
+
+  for (xi1 = xi1mid-rad+xoffset/gridsize; xi1 <= xi1mid+rad+xoffset/gridsize; xi1 += rad / gridsize)
+    for (xi2 = xi2mid-rad+yoffset/gridsize; xi2 <= xi2mid+rad+yoffset/gridsize; xi2 += rad / gridsize)
+      //  for (xi1 = xi1mid-rad; xi1 <= xi1mid+rad; xi1 += rad / gridsize)
+      //    for (xi2 = xi2mid-rad; xi2 <= xi2mid+rad; xi2 += rad / gridsize)
       {
 	Point3d hp = p + xi1 * t1 + xi2 * t2;
 
@@ -2308,7 +2555,7 @@ SetOpenGlColor(double h, double hmin, double hmax, int logscale)
 
   value *= 4;
 
-  const double colp[][3] =
+  static const double colp[][3] =
   {
     { 1, 0, 0 },
     { 1, 1, 0 },
@@ -2323,8 +2570,7 @@ SetOpenGlColor(double h, double hmin, double hmax, int logscale)
   double r = value - i;
 
   GLdouble col[3];
-  int j;
-  for (j = 0; j < 3; j++)
+  for (int j = 0; j < 3; j++)
     col[j] = (1-r) * colp[i][j] + r * colp[i+1][j];
   
   glColor3d (col[0], col[1], col[2]);
@@ -2472,6 +2718,7 @@ DrawClipPlaneTrig (const SolData * sol,
 		   int level)
 {
   int j;
+  double val;
   if (level <= 0)
     for (j = 0; j < 3; j++)
       {
@@ -2494,11 +2741,12 @@ DrawClipPlaneTrig (const SolData * sol,
 	    p += def;
 	  }
 
-	double val = 
-	  GetValue (sol, trig.elnr, 
-		    trig.points[j].lami(0),
-		    trig.points[j].lami(1),
-		    trig.points[j].lami(2), scalcomp);
+	
+	// TODO: consider return value (bool: draw/don't draw element)
+	GetValue (sol, trig.elnr, 
+		  trig.points[j].lami(0),
+		  trig.points[j].lami(1),
+		  trig.points[j].lami(2), scalcomp, val);
       
 	SetOpenGlColor  (val, minval, maxval, logscale);
 	glVertex3dv (p);
@@ -2552,30 +2800,37 @@ int Ng_Vis_Set (ClientData clientData,
 		      "vec") == 0)
 	    vssolution.clipsolution = 2;
 	    
-	  const char * scalname = 
+	  // SZ const -> tcl_const
+	  tcl_const char * scalname =  
 	    Tcl_GetVar (interp, "visoptions.scalfunction", 0);
-	  const char * vecname = 
+	  // SZ const -> tcl_const 
+	  tcl_const char * vecname = 
 	    Tcl_GetVar (interp, "visoptions.vecfunction", 0);
 	  
 	  vssolution.scalfunction = -1;
 	  vssolution.vecfunction = -1;
 
+	  int pointpos; // SZ 
+	  char * pch;
+	  pch=strchr(scalname,'.');
+	  pointpos = int(pch-scalname+1);
+
 	  for (i = 0; i < vssolution.soldata.Size(); i++)
 	    {
 	      if (strlen (vssolution.soldata[i]->name) ==
-		  strlen (scalname)-2 &&
+		  pointpos-1 &&
 		  strncmp (vssolution.soldata[i]->name, scalname,
-			   strlen (scalname)-2) == 0)
+			   pointpos-1) == 0)
 		{
 		  vssolution.scalfunction = i;
-		  vssolution.scalcomp = atoi (scalname + strlen(scalname)-1);
+		  vssolution.scalcomp = atoi (scalname + pointpos);
 		}
 	      if (strcmp (vssolution.soldata[i]->name, vecname) == 0)
 		vssolution.vecfunction = i;
 	    }
 
-
-	  const char * evalname = 
+	  
+	  tcl_const char * evalname = 
 	    Tcl_GetVar (interp, "visoptions.evaluate", 0);
 	  
 	  if (strcmp(evalname, "abs") == 0) vssolution.evalfunc = VisualSceneSolution::FUNC_ABS;
@@ -2586,6 +2841,14 @@ int Ng_Vis_Set (ClientData clientData,
 	  vssolution.gridsize = 
 	    atoi (Tcl_GetVar (interp, "visoptions.gridsize", 0));
 
+	  vssolution.xoffset = 
+	    atof (Tcl_GetVar (interp, "visoptions.xoffset", 0));
+
+	  //	  cout << "x-offset:" << vssolution.xoffset << endl;
+
+	  vssolution.yoffset = 
+	    atof (Tcl_GetVar (interp, "visoptions.yoffset", 0));
+
 	  vssolution.autoscale = 
 	    atoi (Tcl_GetVar (interp, "visoptions.autoscale", 0));
 
@@ -2610,6 +2873,9 @@ int Ng_Vis_Set (ClientData clientData,
 	  vssolution.numtexturecols = 
 	    atoi (Tcl_GetVar (interp, "visoptions.numtexturecols", 0));
 
+	  vssolution.multidimcomponent = 
+	    atoi (Tcl_GetVar (interp, "visoptions.multidimcomponent", 0));
+
 	  vssolution.draw_fieldlines = 
 	    atoi (Tcl_GetVar (interp, "visoptions.drawfieldlines", 0));
 	  vssolution.num_fieldlines = 
@@ -2638,6 +2904,7 @@ int Ng_Vis_Set (ClientData clientData,
       if (argc >= 3 && strcmp (argv[1], "time") == 0)
 	{
 	  vssolution.time = double (atoi (argv[2])) / 1000;
+	  vssolution.solutiontimestamp = NextTimeStamp();
 	  // cout << "time = " << vssolution.time << endl;
 	}
 
diff --git a/Netgen/libsrc/visualization/vssolution.hpp b/Netgen/libsrc/visualization/vssolution.hpp
index 8a63ffd6ba..d20654b5fc 100644
--- a/Netgen/libsrc/visualization/vssolution.hpp
+++ b/Netgen/libsrc/visualization/vssolution.hpp
@@ -36,6 +36,7 @@ class VisualSceneSolution : public VisualScene
   int linelist;
   int clipplanelist;
   int isolinelist;
+  int clipplane_isolinelist;
   int surface_vector_list;
   int cone_list;
 
@@ -46,6 +47,7 @@ class VisualSceneSolution : public VisualScene
   int num_fieldlineslists;
 
   int surfeltimestamp, clipplanetimestamp, solutiontimestamp;
+  int surfellinetimestamp;
   int fieldlinestimestamp, surface_vector_timestamp;
   double minval, maxval;
 
@@ -78,6 +80,9 @@ public:
   class SolData
   {
   public:
+    SolData ();
+    ~SolData ();
+    
     char * name;
     double * data;
     int components;
@@ -101,6 +106,7 @@ public:
   int clipsolution;  // 0..no, 1..scal, 2..vec
   int scalfunction, scalcomp, vecfunction;
   int gridsize;
+  double xoffset, yoffset;
 
   int autoscale, logscale;
   double mminval, mmaxval;
@@ -112,6 +118,8 @@ public:
   bool lineartexture;
   int numtexturecols;
 
+  int multidimcomponent;
+
   // bool fieldlineplot;
   double time;
 
@@ -146,16 +154,16 @@ private:
 
 
   // Get Function Value, local coordinates lam1, lam2, lam3, 
-  double GetValue (const SolData * data, ElementIndex elnr, 
+  bool GetValue (const SolData * data, ElementIndex elnr, 
 		   double lam1, double lam2, double lam3,
-		   int comp) const;
-  double GetSurfValue (const SolData * data, SurfaceElementIndex elnr,
-		       double lam1, double lam2, 
-		       int comp) const;
-  void GetValues (const SolData * data, ElementIndex elnr, 
+		   int comp, double & val) const;
+  bool GetSurfValue (const SolData * data, SurfaceElementIndex elnr,
+		     double lam1, double lam2, 
+		     int comp, double & val) const;
+  bool GetValues (const SolData * data, ElementIndex elnr, 
 		  double lam1, double lam2, double lam3,
 		  double * values) const;
-  void GetSurfValues (const SolData * data, SurfaceElementIndex elnr,
+  bool GetSurfValues (const SolData * data, SurfaceElementIndex elnr,
 		      double lam1, double lam2, 
 		      double * values) const;
 
diff --git a/Netgen/nglib_addon.cpp b/Netgen/nglib_addon.cpp
index 8416c0af55..391e036012 100644
--- a/Netgen/nglib_addon.cpp
+++ b/Netgen/nglib_addon.cpp
@@ -2,13 +2,19 @@
 
 #include "meshing.hpp"
 #include "mystdlib.h"
+
+namespace nglib {
 #include "nglib.h"
+}
 
 using namespace netgen;
 
 #include <iostream>
 #include "Message.h"
 
+namespace nglib
+{
+
 class mystreambuf: public streambuf
 {
   int index;
@@ -73,7 +79,7 @@ Ng_Result NgAddOn_GenerateVolumeMesh (Ng_Mesh * mesh, Ng_Meshing_Parameters * mp
 
   MeshVolume (mparam, *m);
   //RemoveIllegalElements (*m);
-  //OptimizeVolume (mparam, *m, NULL);
+  //OptimizeVolume (mparam, *m);
 
   return NG_OK;
 }
@@ -91,7 +97,9 @@ Ng_Result NgAddOn_OptimizeVolumeMesh (Ng_Mesh * mesh, Ng_Meshing_Parameters * mp
 
   //MeshVolume (mparam, *m);
   RemoveIllegalElements (*m);
-  OptimizeVolume (mparam, *m, NULL);
+  OptimizeVolume (mparam, *m);
 
   return NG_OK;
 }
+
+}
-- 
GitLab