diff --git a/Geo/GFace.h b/Geo/GFace.h
index 2daf19571c65f1b128305a1c5c8c65fafcc466bf..67447ca4e9bf611d20950bb3a98c515817d8f83c 100644
--- a/Geo/GFace.h
+++ b/Geo/GFace.h
@@ -131,8 +131,17 @@ class GFace : public GEntity
   std::vector<MQuadrangle*> quadrangles;
 
   struct {
+    // do we recombine the triangles of the mesh ?
     int    recombine;
+    // what is the treshold angle for recombination
     double recombineAngle;
+    // is this surface meshed using a transfinite interpolation
+    int   Method;
+    // these are the 3 corners of the interpolation
+    std::vector<GVertex* > corners;
+    // all diagonals of the triangulation are left (1), right (2)
+    // or alternated (3)
+    int    transfiniteArrangement;
   } meshAttributes ;
 
 };
diff --git a/Geo/gmshFace.cpp b/Geo/gmshFace.cpp
index 332f886bd60228ef1bb70a7a3e0022bc324f5d11..6622a37dfa343a10c39763693d3764f242cd3f1f 100644
--- a/Geo/gmshFace.cpp
+++ b/Geo/gmshFace.cpp
@@ -1,4 +1,4 @@
-// $Id: gmshFace.cpp,v 1.17 2006-08-29 10:39:48 remacle Exp $
+// $Id: gmshFace.cpp,v 1.18 2006-09-05 21:37:59 remacle Exp $
 //
 // Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
 //
@@ -71,6 +71,20 @@ gmshFace::gmshFace(GModel *m, Surface *face)
 
   meshAttributes.recombine = s->Recombine;
   meshAttributes.recombineAngle = s->RecombineAngle;
+  meshAttributes.Method = s->Method;
+  if (s->Method == TRANSFINI)
+    {
+      meshAttributes.Method = s->Method;
+      meshAttributes.transfiniteArrangement = s->Recombine_Dir;
+      for (int i=0;i< List_Nbr(s->TrsfPoints);i++)
+	{
+	  Vertex *corn;
+	  List_Read(s->TrsfPoints, i, &corn);
+	  GVertex *gv = m->vertexByTag(corn->Num);
+	  if(!gv) throw;
+	  meshAttributes.corners.push_back(gv);
+	}
+    }
 }
 
 gmshFace::gmshFace(GModel *m, int num)
diff --git a/Mesh/Makefile b/Mesh/Makefile
index 76fe131e9839459e579602b360317ed1ee2cd2d5..0e79b693dcc9c3053cf7cc75a34fd0a24270e66a 100644
--- a/Mesh/Makefile
+++ b/Mesh/Makefile
@@ -1,4 +1,4 @@
-# $Id: Makefile,v 1.128 2006-09-03 07:44:10 geuzaine Exp $
+# $Id: Makefile,v 1.129 2006-09-05 21:37:59 remacle Exp $
 #
 # Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
 #
@@ -65,6 +65,7 @@ SRC = 1D_Mesh.cpp \
       Metric.cpp \
       meshGEdge.cpp \
       meshGFace.cpp \
+      meshGFaceTransfinite.cpp \
       meshGRegion.cpp \
       Nurbs.cpp \
       Interpolation.cpp \
diff --git a/Mesh/meshGFace.cpp b/Mesh/meshGFace.cpp
index d7481e218ce1bb5c1a893d2ef430904a1262875c..4f204b1f336f0bc50a0fe9c5490fb69c460e02e4 100644
--- a/Mesh/meshGFace.cpp
+++ b/Mesh/meshGFace.cpp
@@ -1,4 +1,4 @@
-// $Id: meshGFace.cpp,v 1.10 2006-09-01 10:10:05 remacle Exp $
+// $Id: meshGFace.cpp,v 1.11 2006-09-05 21:37:59 remacle Exp $
 //
 // Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
 //
@@ -103,21 +103,18 @@ public :
   }
 };
 
-class fromParametricToCartesian
+fromParametricToCartesian::fromParametricToCartesian ( GFace *_gf )  
+  : gf(_gf)
 {
-  GFace *gf;
-public :
-  fromParametricToCartesian ( GFace *_gf )  
-    : gf(_gf){}
-  void operator () (MVertex * v)
-  {
-    GPoint coords =  gf->point (SPoint2(v->x(),v->y()));
-    v->x() = coords.x();  
-    v->y() = coords.y();
-    v->z() = coords.z();
-  } 
-};
+}
 
+void fromParametricToCartesian::operator () (MVertex * v)
+{
+  GPoint coords =  gf->point (SPoint2(v->x(),v->y()));
+  v->x() = coords.x();  
+  v->y() = coords.y();
+  v->z() = coords.z();
+} 
 
 void computeEdgeLoops (const GFace *gf,
 		       std::vector<MVertex*> & all_mvertices,
@@ -362,7 +359,7 @@ void RefineMesh ( GFace *gf, BDS_Mesh &m , const int NIT)
 	  ++it;
 	}
 
-      Msg(STATUS2," %d triangles : conv %g -> %g (%g sec)",m.triangles.size(),maxL,1.5,(double)(clock()-t1)/CLOCKS_PER_SEC);
+      //      Msg(STATUS2," %d triangles : conv %g -> %g (%g sec)",m.triangles.size(),maxL,1.5,(double)(clock()-t1)/CLOCKS_PER_SEC);
       if ((minL > 0.4 && maxL < 1.5) || IT > NIT)break;
 
 
@@ -1010,6 +1007,9 @@ void meshGFace :: operator() (GFace *gf)
   // Only apply this technique to unknown surfaces or planar surfaces
   // when it is unknown, try your best ...
 
+  if (MeshTransfiniteSurface(gf))return;
+
+
   std::vector<MVertex*> points;
   std::vector<int> indices;
   // compute loops on the fly
diff --git a/Mesh/meshGFace.h b/Mesh/meshGFace.h
index a3c4191f045c2d2da3e6b1d6b312f287807e709e..cfd1401e6d79b238159092aa0dcc8ab81e9f3f9d 100644
--- a/Mesh/meshGFace.h
+++ b/Mesh/meshGFace.h
@@ -20,7 +20,25 @@
 // 
 // Please report all bugs and problems to <gmsh@geuz.org>.
 
+#include <vector>
+class MVertex;
 class GFace;
+// compute edge loops of the face, all_mvertices
+// are the vertices of the 
+void computeEdgeLoops (const GFace *gf,
+		       std::vector<MVertex*> & all_mvertices,
+		       std::vector<int> &indices);
+
+int MeshTransfiniteSurface( GFace *gf);
+
+class fromParametricToCartesian
+{
+  GFace *gf;
+public :
+  fromParametricToCartesian ( GFace *_gf ) ; 
+  void operator () (MVertex * v);
+};    
+
 // Create the mesh of the face
 class meshGFace 
 {
diff --git a/Mesh/meshGFaceTransfinite.cpp b/Mesh/meshGFaceTransfinite.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..121effb6849610c39b23cf2ae599b54017042ea7
--- /dev/null
+++ b/Mesh/meshGFaceTransfinite.cpp
@@ -0,0 +1,226 @@
+// $Id: meshGFaceTransfinite.cpp,v 1.1 2006-09-05 21:37:59 remacle Exp $
+//
+// Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
+//
+// This program is free software; you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation; either version 2 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program; if not, write to the Free Software
+// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
+// USA.
+// 
+// Please report all bugs and problems to <gmsh@geuz.org>.
+
+#include <map>
+#include "meshGFace.h"
+#include "GVertex.h"
+#include "GEdge.h"
+#include "GFace.h"
+#include "MVertex.h"
+#include "MElement.h"
+#include "Context.h"
+#include "Message.h"
+
+#define TRAN_QUA(c1,c2,c3,c4,s1,s2,s3,s4,u,v) \
+   (1.-u)*c4+u*c2+(1.-v)*c1+v*c3-((1.-u)*(1.-v)*s1+u*(1.-v)*s2+u*v*s3+(1.-u)*v*s4)
+
+int MeshTransfiniteSurface( GFace *gf)
+{
+  if(gf->meshAttributes.Method != TRANSFINI)
+    return 0;
+  std::vector <MVertex *> corners;
+  std::vector <MVertex *> d_vertices;
+  std::vector <int> indices;
+
+  for (int i=0;i<gf->meshAttributes.corners.size();i++)
+    corners.push_back(gf->meshAttributes.corners[i]->mesh_vertices[0]);
+
+  computeEdgeLoops (gf,d_vertices, indices);
+
+  if (corners.size () != 3 && corners.size () != 4)
+    Msg (GERROR,"Surface %d is transfinite but has %d corners",gf->tag(),corners.size());
+  if (indices.size () != 2)
+    Msg (GERROR,"Surface %d is transfinite but has %d holes",gf->tag(),indices.size()-2);
+
+  std::vector <MVertex *> m_vertices;
+  int I;
+  for (I=0;I<d_vertices.size();I++)
+    if(d_vertices[I] == corners[0])break;
+  for (int j=0;j<d_vertices.size();j++)
+    m_vertices.push_back(d_vertices[(I+j)%d_vertices.size()]);
+  
+  int iCorner = 0;
+  int N[4];
+  std::vector<double> U;
+  std::vector<double> V;
+
+  std::map<std::pair<int,int> , MVertex*> tab;
+
+  for (int i=0;i<m_vertices.size();i++)
+    {
+      MVertex *v = m_vertices[i];
+      if (v==corners[0]||v==corners[1]||v==corners[2]|| (corners.size()==4 && v==corners[3]))
+	{
+	  N[iCorner++] = i;	  
+	  if (iCorner > 4)
+	    Msg (GERROR,"Surface %d transfinite parameters are incoherent",gf->tag());	  
+	}
+      SPoint2 param =  gf->parFromPoint (SPoint3(v->x(),v->y(),v->z()));      
+      U.push_back(param.x());
+      V.push_back(param.y());
+    }
+
+  int N1 = N[0];
+  int N2 = N[1];
+  int N3 = N[2];
+  int N4 = N[3];
+  
+  int L = N2-N1;
+  int H = N3-N2;
+
+  std::vector<double> lengths_i;
+  std::vector<double> lengths_j;
+
+  double L_i=0;
+  double L_j=0;
+
+  lengths_i.push_back( 0. );
+  lengths_j.push_back( 0. );
+  for (int i=0;i<L;i++)
+    {
+      MVertex *v1 = m_vertices[i];
+      MVertex *v2 = m_vertices[i+1];
+      double l = sqrt ((v1->x()-v2->x())*(v1->x()-v2->x())+
+		       (v1->y()-v2->y())*(v1->y()-v2->y())+
+		       (v1->z()-v2->z())*(v1->z()-v2->z()));
+      L_i+= l;
+      lengths_i.push_back( L_i );
+    }
+  for (int i=L;i<L+H;i++)
+    {
+      MVertex *v1 = m_vertices[i];
+      MVertex *v2 = m_vertices[i+1];
+      double l = sqrt ((v1->x()-v2->x())*(v1->x()-v2->x())+
+		       (v1->y()-v2->y())*(v1->y()-v2->y())+
+		       (v1->z()-v2->z())*(v1->z()-v2->z()));
+      L_j+= l;
+      lengths_j.push_back( L_j );
+    }
+
+
+  //Msg (INFO,"L %d H %d -- %d -- %d %d %d %d",L,H,m_vertices.size(),N1,N2,N3,N4);
+
+  /*
+       2L+H
+         +------------+ L+H
+         |            |
+         |            |
+         |            |
+2L+2H+2  |            |
+         +------------+
+         0             L
+ 
+  */
+
+  tab[std::make_pair(0,0)] = m_vertices[0];
+  tab[std::make_pair(L,0)] = m_vertices[L];
+  tab[std::make_pair(L,H)] = m_vertices[L+H];
+  tab[std::make_pair(0,H)] = m_vertices[2*L+H];
+
+  for (int i=1;i<L;i++)
+    {
+      tab[std::make_pair(i,0)] = m_vertices[i];
+      tab[std::make_pair(i,H)] = m_vertices[2*L+H-i];
+    }
+  for (int i=1;i<H;i++)
+    {
+      tab[std::make_pair(L,i)] = m_vertices[L+i];
+      tab[std::make_pair(0,i)] = m_vertices[2*L+2*H-i];
+    }
+
+  double UC1 = U[N1];
+  double UC2 = U[N2];
+  double UC3 = U[N3];
+  double UC4 = U[N4];
+  double VC1 = V[N1];
+  double VC2 = V[N2];
+  double VC3 = V[N3];
+  double VC4 = V[N4];
+
+  for(int i = 1; i<L; i++) 
+    {
+      double u = lengths_i[i]/L_i;
+      for(int j = 1; j < H; j++) 
+	{
+	  double v = lengths_j[j]/L_j;
+	  int iP1 = N1+i;
+	  int iP2 = N2+j;
+	  int iP3 = N4-i;
+	  int iP4 = (N4+(N3-N2)-j)%m_vertices.size();
+	  
+	  double Up = TRAN_QUA ( U[iP1], U[iP2], U[iP3], U[iP4] , UC1, UC2, UC3, UC4, u, v ); 
+	  double Vp = TRAN_QUA ( V[iP1], V[iP2], V[iP3], V[iP4] , VC1, VC2, VC3, VC4, u, v ); 
+
+	  GPoint gp = gf->point (SPoint2(Up,Vp));
+	  MFaceVertex *newv  = new MFaceVertex ( gp.x(),gp.y(),gp.z(), gf, Up, Vp );
+	  gf->mesh_vertices.push_back (newv);	  	  
+	  tab[std::make_pair(i,j)] = newv;
+	}
+    }
+
+
+  for (int IT = 0;IT< 5;IT++)
+    {
+      for(int i = 1; i<L; i++) 
+	{
+	  for(int j = 1; j < H; j++) 
+	    {
+	      MVertex *v1 = tab[std::make_pair(i,j)];
+	      MVertex *v2 = tab[std::make_pair(i+1,j)];
+	      MVertex *v3 = tab[std::make_pair(i-1,j)];
+	      MVertex *v4 = tab[std::make_pair(i,j+1)];	  
+	      MVertex *v5 = tab[std::make_pair(i,j-1)];	  
+	      
+	      v1->x() = 0.25 * (v2->x() +v3->x() +v4->x() +v5->x());
+	      v1->y() = 0.25 * (v2->y() +v3->y() +v4->y() +v5->y());
+	      v1->z() = 0.25 * (v2->z() +v3->z() +v4->z() +v5->z());
+	    }
+	}
+    }
+
+  for(int i = 0; i < L ; i++) 
+    {
+      for(int j = 0; j < H; j++) 
+	{
+	  MVertex *v1 = tab[std::make_pair(i,j)];
+	  MVertex *v2 = tab[std::make_pair(i+1,j)];
+	  MVertex *v3 = tab[std::make_pair(i+1,j+1)];
+	  MVertex *v4 = tab[std::make_pair(i,j+1)];
+	  if (gf->meshAttributes.recombine)
+	    gf->quadrangles.push_back(new MQuadrangle (v1,v2,v3,v4));
+	  else if (  gf->meshAttributes.transfiniteArrangement == 1 ||
+		     (gf->meshAttributes.transfiniteArrangement == 0 && 
+		      (( i % 2 == 0 && j % 2 == 1 ) ||
+		       ( i % 2 == 1 && j % 2 == 0 ) ) ) )
+	    {
+	      gf->triangles.push_back(new MTriangle (v1,v2,v3));
+	      gf->triangles.push_back(new MTriangle (v3,v4,v1));
+	    }	  
+	  else 
+	    {
+	      gf->triangles.push_back(new MTriangle (v1,v2,v4));
+	      gf->triangles.push_back(new MTriangle (v4,v2,v3));
+	    }	  
+	}
+    }
+  return 1;
+}
+  
diff --git a/Mesh/meshGRegion.cpp b/Mesh/meshGRegion.cpp
index 42ea7dc5f595b07a0160215f385c15065261a2f0..1c27b806c9bb443656baf554d6946a0ebc230843 100644
--- a/Mesh/meshGRegion.cpp
+++ b/Mesh/meshGRegion.cpp
@@ -204,6 +204,12 @@ void meshNormalsPointOutOfTheRegion (GRegion *gr)
 	  double v2[3] = {X[2]-X[1],Y[2]-Y[1],Z[2]-Z[1]};
 	  double N[3] ;
 	  prodve ( v1, v2 , N );
+	  norme (v1);
+	  norme (v2);
+	  norme(N);
+	  N[0]+= 0.05*v1[0]-0.05*v2[0];
+	  N[1]+= 0.03*v1[1]-0.15*v2[1];
+	  N[2]+= 0.05*v1[2]-0.05*v2[2];
 	  norme(N);
 	  std::list<GFace*>::iterator it_b = faces.begin();
 	  while (it_b != faces.end())
diff --git a/benchmarks/2d/transfinite_b.geo b/benchmarks/2d/transfinite_b.geo
new file mode 100644
index 0000000000000000000000000000000000000000..8efc8c45351a05dde8707afd9f3250c640ebb243
--- /dev/null
+++ b/benchmarks/2d/transfinite_b.geo
@@ -0,0 +1,19 @@
+Point(1) = {0,0,0,0.1};
+Point(2) = {1,0,0,0.1};
+Point(3) = {0,1,0,0.1};
+Point(4) = {3,0,0,0.1};
+Point(5) = {3,3,0,0.1};
+Point(6) = {0,3,0,0.1};
+Line(1) = {3,6};
+Line(2) = {6,5};
+Line(3) = {5,4};
+Line(4) = {4,2};
+Circle(5) = {2,1,3};
+Line Loop(6) = {3,4,5,1,2};
+Plane Surface(7) = {6};
+Transfinite Line {1} = 20 Using Progression 1.2;
+Transfinite Line {4} = 20 Using Progression 1./1.2;
+Transfinite Line {3,2} = 10 Using Progression 1;
+Transfinite Line {5} = 19 Using Progression 1;
+Transfinite Surface {7} = {3,6,4,2} Alternated;
+Recombine Surface {7};