diff src/cs/drivers/drv_app/r2d/r2d.c @ 0:92470e5d0b9e

src: partial import from FC Selenite
author Mychaela Falconia <falcon@freecalypso.org>
date Fri, 15 May 2020 01:28:16 +0000
parents
children
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/cs/drivers/drv_app/r2d/r2d.c	Fri May 15 01:28:16 2020 +0000
@@ -0,0 +1,5365 @@
+/**
+                                                                          
+  @file       r2d.c   
+  
+  @author     Christophe Favergeon
+
+  @version    0.5
+                                                                          
+  Function    Riviera 2D system implementation
+              (higher software layer)                       
+     
+*/
+
+/*
+  Date        Modification                                                 
+  ------------------------                                                 
+  06/12/2001    Create    
+  10/18/2001    Version 0.5 for first integration with Riviera database
+ **************************************************************************
+  History       
+                                                                          
+  Apr-06-2005 MMI-SPR-29655 - xreddymn
+  Added function r2d_flush_region to allow flushing of portions 
+  of framebuffer to the display
+
+*/
+
+ 
+#include "rv/general.h"
+#include "rvf/rvf_api.h"
+#include "r2d/r2d_config.h"
+#include "r2d/r2d.h"
+#include "r2d/r2d_i.h"
+#include "r2d/r2d_independance_layer.h"
+#include "r2d/r2d_convertion_table.h"
+#include "r2d/r2d_env.h"
+#include "rvm/rvm_use_id_list.h"
+#include <string.h>
+
+#include "r2d_cos.c"
+
+
+#ifdef _WINDOWS
+  #pragma warning(disable:4761) // integral size mismatch in argument; conversion supplied
+#endif
+
+
+/***************************************
+
+     IMPORTANT INFORMATION
+
+***************************************/
+
+/*
+
+Clipping is done on original coordinates.
+Then those coordinates are modified to take into account a possible
+mirror symetry of the LCD. 
+Then the scanning algorithm is applied on those transformed and clipped
+coordinates. It assumes a vertical scanning (column per column).
+If it is not the case. The high level coordinates (x,y) are swapped
+after clipping and mirroring. The scanning is done on the swapped coordinates
+so that coulmns are mapped to lines.
+
+Subroutine's comment use following rule:
+H -> High level user coordinates
+L -> LCD coordinates (mirroring or not and local_to_global applied)
+S -> Scanning coordinates (vertical or horizontal)
+*/
+
+/***************************************
+
+     CONSTANTS
+
+***************************************/
+
+// Used for char drawing algorithm with unicode
+typedef enum K_R2D_CHAR_DRAWING_MODE
+{
+R2D_COMBINING_CHAR=1
+} T_R2D_CHAR_DRAWING_MODE;
+
+// Used for internal drawing function
+typedef enum K_R2D_TEXT_DRAWING_SETTINGS
+{
+  R2D_TEXT_DRAW=1
+} T_R2D_TEXT_DRAWING_SETTINGS;
+
+
+
+
+
+/***************************************
+
+     REFRESH TASK RELATED GLOBALS
+
+***************************************/
+
+extern INT16 r2d_update_ul_x,r2d_update_ul_y,r2d_update_br_x,r2d_update_br_y;
+
+/***************************************
+
+     STATIC INTERNAL FUNCTIONS
+	 AND GLOBALS
+
+***************************************/
+
+
+#define IND_r2d_write_lcd_line r2d_write_lcd_line
+#define IND_r2d_write_lcd_pixel r2d_write_lcd_pixel
+#define IND_r2d_blit_lcd_to_lcd r2d_blit_lcd_to_lcd
+#define IND_r2d_blit_lcd_to_color r2d_blit_lcd_to_color
+#define IND_r2d_blit_color_to_lcd r2d_blit_color_to_lcd
+#define IND_r2d_blit_color_to_color r2d_blit_color_to_color
+#define IND_r2d_get_color_pixel_value r2d_get_color_pixel_value
+
+static T_R2D_GC_PTR r2d_new_font_buffer_context(T_RVF_MB_ID bank,
+												T_R2D_FRAMEBUFFER_KIND kind,
+												UINT16 max_width,
+												UINT16 ascent);
+static T_R2D_FONT_CACHE_FRAMEBUFFER* r2d_new_font_framebuffer(T_RVF_MB_ID bank,
+															  T_R2D_FRAMEBUFFER_KIND the_kind,UINT16 width, UINT16 height);
+static T_R2D_SHAPE_PTR r2d_s_level(T_R2D_GC_PTR gc,T_R2D_SHAPE_PTR self);
+static void r2d_mirror_rectangle(T_R2D_GC_PTR gc,T_R2D_SHAPE_PTR rectangle);
+static void r2d_mirror_clip_rectangle(T_R2D_GC_PTR gc,T_R2D_SHAPE_PTR rectangle);
+static void r2d_diagonal_mirror(T_R2D_SHAPE_PTR rectangle);
+
+static void r2d_update_region(T_R2D_GC *gc,INT16 ul_x,INT16 ul_y,INT16 br_x,INT16 br_y);
+
+void r2d_translate_shape(T_R2D_SHAPE_PTR self,INT16 dx,INT16 dy);
+
+// Used for pen_size != 1. It computes the bounding rect and calls
+// fillrectangle
+static void r2d_s_fill_point(T_R2D_GC_PTR gc,INT16 x,INT16 y,INT16 pen_size);
+
+static void           r2d_df_rectangle(T_R2D_GC_PTR gc,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y,BOOLEAN background);
+static void           r2d_s_fill_rectangle(T_R2D_GC_PTR gc,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y,BOOLEAN background);
+
+
+
+#define R2D_SQUARE_P(x) (x*x)
+#define R2D_SWAP(x,y) tmp=x; x=y; y=tmp
+#define SQRT2 0x16A09
+#define PI 0x3243F
+
+////////////////////////////////////////
+//
+// Utility functions
+//
+
+// L COORDINATES
+void r2d_update_region(T_R2D_GC* gc,INT16 ul_x,INT16 ul_y,INT16 br_x,INT16 br_y)
+{
+
+	INT16 tmp;
+
+	rvf_lock_mutex(r2d_g_global_mutex);
+
+	if (ul_x>br_x)
+	{
+		tmp=ul_x;
+		ul_x=br_x;
+		br_x=tmp;
+	}
+
+	if (ul_y>br_y)
+	{
+		tmp=ul_y;
+		ul_y=br_y;
+		br_y=tmp;
+	}
+
+	if (gc)
+	{
+	  if (ul_x<((T_R2D_RECT*)(gc->p_clipping_shape))->ul_x-gc->org_x)
+         ul_x=((T_R2D_RECT*)(gc->p_clipping_shape))->ul_x-gc->org_x;
+      if (ul_y<((T_R2D_RECT*)(gc->p_clipping_shape))->ul_y-gc->org_y)
+         ul_y=((T_R2D_RECT*)(gc->p_clipping_shape))->ul_y-gc->org_y;
+      if (br_x>=((T_R2D_RECT*)(gc->p_clipping_shape))->br_x-1-gc->org_x)
+         br_x=((T_R2D_RECT*)(gc->p_clipping_shape))->br_x-1-gc->org_x;
+      if (br_y>=((T_R2D_RECT*)(gc->p_clipping_shape))->br_y-1-gc->org_y)
+         br_y=((T_R2D_RECT*)(gc->p_clipping_shape))->br_y-1-gc->org_y;
+    }
+
+	if (ul_x < r2d_update_ul_x)
+		r2d_update_ul_x=ul_x;
+    if (ul_y < r2d_update_ul_y)
+		r2d_update_ul_y=ul_y;
+	if (br_x > r2d_update_br_x)
+		r2d_update_br_x=br_x;
+	if (br_y > r2d_update_br_y)
+		r2d_update_br_y=br_y;
+
+	
+	  if (r2d_update_ul_x<0)
+         r2d_update_ul_x=0;
+      if (r2d_update_ul_y<0)
+         r2d_update_ul_y=0;
+      if (r2d_update_br_x>=R2D_WIDTH)
+         r2d_update_br_x=R2D_WIDTH-1;
+      if (r2d_update_br_y>=R2D_HEIGHT)
+         r2d_update_br_y=R2D_HEIGHT-1;
+	
+	
+	rvf_unlock_mutex(r2d_g_global_mutex);
+}
+
+
+void r2d_check_and_send_event(T_R2D_GC_PTR gc)
+{
+	  // If routine has drawn into the LCD framebuffer
+	  // then refresh may be required
+	  // If LCD framebuffer ONLY
+	  if (((T_R2D_FRAMEBUFFER*)(((T_R2D_GC*)gc)->p_frame_buffer))->kind==0)
+	  {
+	  if ((r2d_g_event_was_sent==FALSE) && (r2d_g_refresh_disabled==0))
+      {
+		  //rvf_send_trace("R2D SEND EVENT",strlen("R2D SEND EVENT"), NULL_PARAM, 
+			   //RV_TRACE_LEVEL_DEBUG_HIGH, TRACE_XXX );
+		     r2d_g_event_was_sent=TRUE;
+			 rvf_send_event( (UINT8)(r2d_addr_id & 0xFF), EVENT_MASK(RVF_APPL_EVT_0) );
+	  }
+	  }
+}
+
+// WILL HAVE TO BE IMPROVED
+
+INT32 r2d_fixed_division(INT32 a,INT32 b)
+{
+    INT16 k;
+    UINT32 q,nb;
+    INT16 right_shift=0;
+    INT16 sign=0;
+
+	// Convert to positive values
+    if ((b<0) && (a<0))
+	{
+		b=-b;
+		a=-a;
+	}
+	if ((b<0) && (a>=0))
+	{
+		b=-b;
+		sign=1;
+	}
+	if ((a<0) && (b>=0))
+	{
+		a=-a;
+		sign=1;
+	}
+
+
+    // Normalize b between in [0.5,1[
+    
+
+    nb=b;
+
+    if (nb>=0x10000)
+    {
+      //printf("%08X/%08X\n",a,nb);
+      while(((nb>>16)&0x0FFFF)!=0)
+      {
+         nb>>=1;
+         right_shift++;
+         //printf("%08X\n",nb);
+      }
+    } else if (nb<0x8000)
+    {
+      while((nb&0x8000)==0)
+      {
+         nb<<=1;
+         right_shift--;
+         //printf("%08X\n",nb);
+      }
+    }
+    //printf("right_shift=%d\n",right_shift);
+    if (nb==0x8000)
+    {
+      //printf("nb=0x8000, return %08X\n",((a>>right_shift)<<1));
+      if (sign)
+        return (-((a>>right_shift)<<1));
+      else
+        return (((a>>right_shift)<<1));
+    }
+
+
+    
+
+    // Newton method used to compute 1/nb with
+    // q=q(2-nb * q)
+
+    // Which first choice for q and which accuracies for intermediate
+    // computations ?
+
+    
+    // nb being in ]0.5,1[ one must has q in ]0.5, 3[
+    // And one must check that the function is contracting
+    // Derivative is nul for q*nb = 1. Below, sign of
+    // derivatuve (relatively to q) is positive if a<1 and
+    // the derivative is between 2 and 0
+    // <1 is got for q>1/(2 nb) and q must be < 1/nb
+    // One can take q=1 as starting value. It solves
+    // for the bounds except for nb=1 or nb=0.5
+
+    // if q in ]1/2nb, 1/nb[ one can check that qnew
+    // will at most reach 3/2nb at next iteration and then will
+    // fall back into the intervall
+    // So, one needs to represents values up to 3 for q
+    // and 2 bits are required for integer part
+    
+    // So, q is 2.16, nb is 0.16
+    // q*nb is 2.32 which cannot be represented and one needs
+    // to shift value by 1 bit
+    // nb will be 0.15 and q 2.15 sor nb*q is 2.30
+    // It must be multiplied by a format 2.16
+    // so both intermediate result must be shifted to
+    // 2.15 (1 for q and 15 for np*q)
+    
+    
+
+    q=(1<<16);
+    //printf("-- q=%08X\n",q);
+
+    
+    for(k=0;k<5;k++)
+    {
+       q=((q>>1) * ((2 << 15) - (((nb>>1) * (q>>1))  >> 15))) >> 14;
+       //printf("-- q=%08X\n",q);
+    }
+    
+    // a is known to be an integer and q is know to be less than 1
+    // so a 64 bits product is not required
+
+    //printf("a=%08X, q=%08X\n",a,q);
+
+    q>>=right_shift;
+
+    q=((a>>16)*q) ;
+
+
+    //printf("right_shift=%d, q=%08X\n",right_shift,q);
+
+    if (sign)
+     return(-((INT32)q));
+    else
+     return(q);
+
+    
+}
+
+////////////////////////////////////////
+//
+// Texture
+//
+
+T_R2D_FREE_TEXTURE_PTR r2d_new_free_texture(T_RVF_MB_ID bank,INT16 size,T_R2D_ARGB_COLOR *pattern)
+{
+   T_R2D_FREE_TEXTURE *t;
+
+   R2D_MALLOC(bank,T_R2D_FREE_TEXTURE,sizeof(T_R2D_FREE_TEXTURE),t);
+
+   if (t)
+   {
+	   t->refcount=1;
+	   t->size=size;
+	   t->pattern=pattern;
+   }
+
+   return(t);
+}
+
+T_R2D_FREE_TEXTURE_PTR r2d_new_const_free_texture(T_RVF_MB_ID bank,INT16 size,T_R2D_ARGB_COLOR *pattern)
+{
+   T_R2D_FREE_TEXTURE *t;
+
+   R2D_MALLOC(bank,T_R2D_FREE_TEXTURE,sizeof(T_R2D_FREE_TEXTURE),t);
+
+   if (t)
+   {
+	   t->refcount=-1;
+	   t->size=size;
+	   t->pattern=pattern;
+   }
+
+   return(t);
+}
+
+void              r2d_release_free_texture(T_R2D_FREE_TEXTURE_PTR texture)
+{
+	if (texture)
+    {
+      // Remove the array of words if needed
+      // (refcount must be 1 else the array should not be deleted)
+      if (R2D_REFCOUNT(texture)==1) 
+	  {
+		if (((T_R2D_FREE_TEXTURE*)texture)->pattern)
+          R2D_FREE(((T_R2D_FREE_TEXTURE*)texture)->pattern);
+	  }
+    }
+    // Autorelease
+    r2d_release(texture);
+}
+
+T_R2D_ANCHORED_TEXTURE_PTR r2d_new_anchored_texture(T_RVF_MB_ID bank,T_R2D_GC_PTR gc,T_R2D_FREE_TEXTURE_PTR texture)
+{
+
+  INT16 length;
+  T_R2D_FREE_TEXTURE *ft;
+  T_R2D_ANCHORED_TEXTURE *at;
+  INT16 i,j,x,y,ni,nj,pos;
+  UINT32 *p;
+
+		 T_R2D_ARGB_COLOR b;
+
+   // If a new texture is installed,
+   // one must recompute p_background
+   if (texture==NULL)
+	  goto erroranchored;
+
+   
+     
+   ft=(T_R2D_FREE_TEXTURE *)texture;
+
+   length=1<<(ft->size);
+
+   R2D_MALLOC(bank,T_R2D_ANCHORED_TEXTURE,sizeof(T_R2D_ANCHORED_TEXTURE),at);
+
+   if (at==NULL)
+	   goto erroranchored;
+
+   at->refcount=1;
+
+   at->size=ft->size;
+
+   R2D_MALLOC(bank,UINT32,sizeof(UINT32)*length*length,at->pattern);
+
+   if (at->pattern==NULL)
+   {
+	   R2D_FREE(at);
+	   goto erroranchored;
+   }
+     
+	 
+   p=at->pattern;
+   b=r2d_get_background_color(gc);
+
+   if (((T_R2D_FRAMEBUFFER*)(((T_R2D_GC*)gc)->p_frame_buffer))->kind==R2D_FULL_KIND)
+   {
+			 x=0;
+			 y=0;
+			 for(i=0;i<length;i++) // Horizontal
+				 for(j=0;j<length;j++) // Vertical
+			 {
+			   r2d_convert_background_color_color(gc,ft->pattern[i*length+j]);
+
+               #if (R2D_REFRESH == R2D_VERTICAL)
+                 #ifdef R2D_MIRRORED_X
+			       ni=(length-1)-i;
+                 #else
+				   ni=i;
+                 #endif
+
+				 #ifdef R2D_MIRRORED_Y
+			       nj=(length-1)-j;
+                 #else
+				   nj=j;
+                 #endif
+                 p[ni*length+nj]=((T_R2D_GC*)gc)->background_pixel_value;
+               #else
+				 #ifdef R2D_MIRRORED_X
+			       nj=(length-1)-j;
+                 #else
+				   nj=j;
+                 #endif
+
+				 #ifdef R2D_MIRRORED_Y
+			       ni=(length-1)-i;
+                 #else
+				   ni=i;
+                 #endif
+
+				 p[nj*length+ni]=((T_R2D_GC*)gc)->background_pixel_value;
+               #endif
+               
+			 }
+	}
+	else
+	{
+			 for(i=0;i<length;i++)
+				 for(j=0;j<length;j++)
+			 {
+			  r2d_convert_background_color(gc,ft->pattern[i*length+j]);
+			  #if (R2D_REFRESH == R2D_VERTICAL)
+                 #ifdef R2D_MIRRORED_X
+			       ni=(length-1)-i;
+                 #else
+				   ni=i;
+                 #endif
+
+				 #ifdef R2D_MIRRORED_Y
+			       nj=(length-1)-j;
+                 #else
+				   nj=j;
+                 #endif
+                 pos=ni*length+nj;
+               #else
+				 #ifdef R2D_MIRRORED_X
+			       nj=(length-1)-j;
+                 #else
+				   nj=j;
+                 #endif
+
+				 #ifdef R2D_MIRRORED_Y
+			       ni=(length-1)-i;
+                 #else
+				   ni=i;
+                 #endif
+
+				 pos=nj*length+ni;
+               #endif
+			  #if (R2D_DITHERING == R2D_OFF)
+                  p[pos]=((T_R2D_GC*)gc)->background_pixel_value;
+              #else
+				  p[pos]=r2d_get_dithering_matrix_entry(((T_R2D_GC*)gc)->p_background_dithered_cache,x,y);      
+              #endif
+
+              #if (R2D_REFRESH == R2D_VERTICAL)
+			    y++;
+			    if (y==length)
+				{
+				  y=0;
+				  x++;
+				}
+              #else
+				x++;
+			    if (x==length)
+				{
+				  x=0;
+				  y++;
+				}
+              #endif
+			 }
+	}
+		 // Force change
+	r2d_set_background_color(gc,b^1);
+    r2d_set_background_color(gc,b);
+		 
+   return((T_R2D_ANCHORED_TEXTURE_PTR)at);
+erroranchored: return(NULL);
+}
+
+void              r2d_release_anchored_texture(T_R2D_ANCHORED_TEXTURE_PTR texture)
+{
+	if (texture)
+    {
+      // Remove the array of words if needed
+      // (refcount must be 1 else the array should not be deleted)
+      if (R2D_REFCOUNT(texture)==1) 
+	  {
+		if (((T_R2D_ANCHORED_TEXTURE*)texture)->pattern)
+          R2D_FREE(((T_R2D_ANCHORED_TEXTURE*)texture)->pattern);
+	  }
+    }
+    // Autorelease
+    r2d_release(texture);
+}
+
+
+////////////////////////////////////////
+//
+// Framebuffer
+//
+
+UINT32 r2d_get_framebuffer_size(T_R2D_FRAMEBUFFER_KIND the_kind,UINT16 width, UINT16 height)
+{
+	UINT32 t;
+	UINT32 length;
+
+	t=0;
+
+	
+			
+
+	t+=sizeof(T_R2D_FRAMEBUFFER);
+		
+	switch(the_kind) 
+	{
+       case R2D_LCD_KIND:
+		    #if (R2D_REFRESH==R2D_VERTICAL)
+              // Column is contiguous in memory
+              length=width*R2D_ALIGNED_MWLENGTH(height);
+              t+=((length+1)<<R2D_LONGSIZE_FACTOR);
+            #else
+			  length=height*R2D_ALIGNED_MWLENGTH(width);
+              t+=((length+1)<<R2D_LONGSIZE_FACTOR);
+            #endif
+	   break;
+	   default:
+			  length=width*height;
+              t+=((length+1)<<R2D_LONGSIZE_FACTOR);
+	   break;
+	}
+
+	t+=sizeof(T_RVF_MUTEX);
+	
+
+	return(t);
+}
+
+UINT32 r2d_get_gc_size(void)
+{
+	UINT32 t;
+
+	t=0;
+
+	
+	t+=sizeof(T_R2D_GC);
+	t+=sizeof(T_R2D_RECT); // clipping
+	t+=sizeof(T_R2D_RECT); // s_clipping
+	t+=sizeof(UINT32)*4; // foreground dithering cache
+	t+=sizeof(UINT32)*4; // backround dithering cache
+	
+
+	return(t);
+}
+
+
+T_R2D_FRAMEBUFFER_PTR r2d_new_framebuffer(T_RVF_MB_ID bank,T_R2D_FRAMEBUFFER_KIND the_kind,UINT16 width, UINT16 height)
+{
+   UINT32 length;
+   T_R2D_FRAMEBUFFER *p;
+//   UINT32 *elem;
+   T_RVF_RET err;
+
+   R2D_MALLOC(bank,T_R2D_FRAMEBUFFER,sizeof(T_R2D_FRAMEBUFFER),p);
+
+   if (p)
+   {
+     switch(the_kind) 
+     {
+        case R2D_LCD_KIND:
+          #if (R2D_REFRESH==R2D_VERTICAL)
+            // Column is contiguous in memory
+            length=width*R2D_ALIGNED_MWLENGTH(height);
+            R2D_MALLOC(bank,UINT32,((length+1)<<R2D_LONGSIZE_FACTOR),p->p_memory_words);
+			if (!(p->p_memory_words))
+			{
+		        R2D_FREE(p);
+		        p=NULL;
+				return(p);
+			}
+			p->refcount=1;
+            p->kind=R2D_LCD_KIND;
+            p->p_frame_buffer_end=p->p_memory_words+length;
+            p->width=width;
+            p->height=height;
+			R2D_MALLOC(bank,T_RVF_MUTEX,sizeof(T_RVF_MUTEX),p->mutex);
+		    if (p->mutex!=NULL)
+			{
+			  err=rvf_initialize_mutex(p->mutex);
+			  if (err!=RVF_OK)
+			  {
+				 R2D_FREE(p->mutex);
+				 R2D_FREE(p->p_memory_words);
+				 R2D_FREE(p);
+				 p=NULL;
+				 return(p);
+			  }
+			}
+			else
+			{
+				R2D_FREE(p->p_memory_words);
+				R2D_FREE(p);
+				p=NULL;
+				return(p);
+			}
+			
+         #else
+             // Line is contiguous in memory
+             length=height*R2D_ALIGNED_MWLENGTH(width);
+             R2D_MALLOC(bank,UINT32,((length+1)<<R2D_LONGSIZE_FACTOR),p->p_memory_words);
+             if (!(p->p_memory_words))
+			{
+		        R2D_FREE(p);
+		        p=NULL;
+				return(p);
+			}
+			 p->refcount=1;
+             p->kind=R2D_LCD_KIND;
+             p->p_frame_buffer_end=p->p_memory_words+length;
+             p->width=width;
+             p->height=height;
+			 R2D_MALLOC(bank,T_RVF_MUTEX,sizeof(T_RVF_MUTEX),p->mutex);
+		    if (p->mutex!=NULL)
+			{
+			  err=rvf_initialize_mutex(p->mutex);
+			  if (err!=RVF_OK)
+			  {
+				 R2D_FREE(p->mutex);
+				 R2D_FREE(p->p_memory_words);
+				 R2D_FREE(p);
+				 p=NULL;
+				 return(p);
+			  }
+			}
+			else
+			{
+				R2D_FREE(p->p_memory_words);
+				R2D_FREE(p);
+				p=NULL;
+				return(p);
+			}
+         #endif
+        break;
+        default:
+            length=width*height;
+            R2D_MALLOC(bank,UINT32,((length+1)<<R2D_LONGSIZE_FACTOR),p->p_memory_words);
+            if (!(p->p_memory_words))
+			{
+		        R2D_FREE(p);
+		        p=NULL;
+				return(p);
+			}
+			p->refcount=1;
+            p->kind=R2D_FULL_KIND;
+            p->p_frame_buffer_end=p->p_memory_words+length;
+            p->width=width;
+            p->height=height;
+			R2D_MALLOC(bank,T_RVF_MUTEX,sizeof(T_RVF_MUTEX),p->mutex);
+		    if (p->mutex!=NULL)
+			{
+			  err=rvf_initialize_mutex(p->mutex);
+			  if (err!=RVF_OK)
+			  {
+				 R2D_FREE(p->mutex);
+				 R2D_FREE(p->p_memory_words);
+				 R2D_FREE(p);
+				 p=NULL;
+				 return(p);
+			  }
+			}
+			else
+			{
+				R2D_FREE(p->p_memory_words);
+				R2D_FREE(p);
+				p=NULL;
+				return(p);
+			}
+        break;
+     }
+   } else p=NULL;
+
+   return(p);
+}
+
+static T_R2D_FONT_CACHE_FRAMEBUFFER* r2d_new_font_framebuffer(T_RVF_MB_ID bank,T_R2D_FRAMEBUFFER_KIND the_kind,UINT16 width, UINT16 height)
+{
+   UINT32 length;
+   T_R2D_FONT_CACHE_FRAMEBUFFER *p;
+//   UINT32 *elem;
+   T_RVF_RET err;
+
+   R2D_MALLOC(bank,T_R2D_FONT_CACHE_FRAMEBUFFER,sizeof(T_R2D_FONT_CACHE_FRAMEBUFFER),p);
+
+   if (p)
+   {
+     switch(the_kind) 
+     {
+        case R2D_LCD_KIND:
+          #if (R2D_REFRESH==R2D_VERTICAL)
+            // Column is contiguous in memory
+            length=width*R2D_ALIGNED_MWLENGTH(height);
+            R2D_MALLOC(bank,UINT32,((length+1)<<R2D_LONGSIZE_FACTOR),p->p_memory_words);
+			if (!(p->p_memory_words))
+			{
+		        R2D_FREE(p);
+		        p=NULL;
+				return(p);
+			}
+            p->refcount=1;
+            p->kind=R2D_LCD_KIND;
+            p->p_frame_buffer_end=p->p_memory_words+length;
+            p->width=width;
+            p->height=height;
+			R2D_MALLOC(bank,T_RVF_MUTEX,sizeof(T_RVF_MUTEX),p->mutex);
+		    if (p->mutex!=NULL)
+			{
+			  err=rvf_initialize_mutex(p->mutex);
+			  if (err!=RVF_OK)
+			  {
+				 R2D_FREE(p->mutex);
+				 R2D_FREE(p->p_memory_words);
+				 R2D_FREE(p);
+				 p=NULL;
+				 return(p);
+			  }
+			}
+			else
+			{
+				R2D_FREE(p->p_memory_words);
+				R2D_FREE(p);
+				p=NULL;
+				return(p);
+			}
+         #else
+             // Line is contiguous in memory
+             length=height*R2D_ALIGNED_MWLENGTH(width);
+             R2D_MALLOC(bank,UINT32,((length+1)<<R2D_LONGSIZE_FACTOR),p->p_memory_words);
+             if (!(p->p_memory_words))
+			{
+		        R2D_FREE(p);
+		        p=NULL;
+				return(p);
+			}
+			 p->refcount=1;
+             p->kind=R2D_LCD_KIND;
+             p->p_frame_buffer_end=p->p_memory_words+length;
+             p->width=width;
+             p->height=height;
+			 R2D_MALLOC(bank,T_RVF_MUTEX,sizeof(T_RVF_MUTEX),p->mutex);
+		    if (p->mutex!=NULL)
+			{
+			  err=rvf_initialize_mutex(p->mutex);
+			  if (err!=RVF_OK)
+			  {
+				 R2D_FREE(p->mutex);
+				 R2D_FREE(p->p_memory_words);
+				 R2D_FREE(p);
+				 p=NULL;
+				 return(p);
+			  }
+			}
+			else
+			{
+				R2D_FREE(p->p_memory_words);
+				R2D_FREE(p);
+				p=NULL;
+				return(p);
+			}
+         #endif
+        break;
+        default:
+            length=width*height;
+            R2D_MALLOC(bank,UINT32,((length+1)<<R2D_LONGSIZE_FACTOR),p->p_memory_words);
+            if (!(p->p_memory_words))
+			{
+		        R2D_FREE(p);
+		        p=NULL;
+				return(p);
+			}
+			p->refcount=1;
+            p->kind=R2D_FULL_KIND;
+            p->p_frame_buffer_end=p->p_memory_words+length;
+            p->width=width;
+            p->height=height;
+			R2D_MALLOC(bank,T_RVF_MUTEX,sizeof(T_RVF_MUTEX),p->mutex);
+		    if (p->mutex!=NULL)
+			{
+			  err=rvf_initialize_mutex(p->mutex);
+			  if (err!=RVF_OK)
+			  {
+				 R2D_FREE(p->mutex);
+				 R2D_FREE(p->p_memory_words);
+				 R2D_FREE(p);
+				 p=NULL;
+				 return(p);
+			  }
+			}
+			else
+			{
+				R2D_FREE(p->p_memory_words);
+				R2D_FREE(p);
+				p=NULL;
+				return(p);
+			}
+        break;
+     }
+     
+   } else p=NULL;
+
+   return(p);
+}
+
+
+// Release a framebuffer
+void r2d_release_framebuffer(T_R2D_FRAMEBUFFER_PTR p)
+{
+    if (p)
+    {
+      // Remove the array of words if needed
+      // (refcount must be 1 else the array should not be deleted)
+      if (R2D_REFCOUNT(p)==1) 
+	  {
+        R2D_FREE(((T_R2D_FRAMEBUFFER*)p)->p_memory_words);
+		rvf_delete_mutex(((T_R2D_FRAMEBUFFER*)p)->mutex);
+		R2D_FREE(((T_R2D_FRAMEBUFFER*)p)->mutex);
+	  }
+    }
+    // Autorelease
+    r2d_release(p);
+}
+
+
+////////////////////////////////////////
+//
+// Graphic context
+//
+
+
+static T_R2D_GC_PTR r2d_new_font_buffer_context(T_RVF_MB_ID bank,
+												T_R2D_FRAMEBUFFER_KIND kind,
+												UINT16 max_width,
+												UINT16 ascent)
+{
+   T_R2D_FRAMEBUFFER_PTR cache_buffer;
+   T_R2D_GC_PTR result;
+
+   //printf("width=%d,height=%d\n",the_width,((T_R2D_FRAMEBUFFER*)font_frame_buffer)->height);
+
+   cache_buffer=r2d_new_font_framebuffer(bank,kind,
+   max_width,ascent);
+
+   
+
+   if (cache_buffer!=NULL)
+   {
+     result=r2d_new_context(bank,cache_buffer);
+     r2d_release_framebuffer(cache_buffer);
+     
+     return(result);
+   } else return(NULL);
+}
+
+
+T_R2D_GC_PTR   r2d_new_context(T_RVF_MB_ID bank,T_R2D_FRAMEBUFFER_PTR the_frame_buffer)
+{
+   T_R2D_GC *gc;
+
+   // Allocate a new gc structure
+   R2D_MALLOC(bank,T_R2D_GC,sizeof(T_R2D_GC),gc);
+   if (gc)
+   {
+      // gc is pointing to the LCD framebuffer
+      gc->p_frame_buffer=(T_R2D_FRAMEBUFFER*)the_frame_buffer;
+      r2d_retain(the_frame_buffer);
+
+      // Default initialization for a gc
+	  gc->background_texture=NULL;
+      gc->foreground_color=r2d_get_standard_argb_color(R2D_BLACK);
+      gc->background_color=r2d_get_standard_argb_color(R2D_WHITE);
+      
+
+	  #if (R2D_REFRESH == R2D_VERTICAL)
+		gc->p_s_clipping_shape=r2d_new_rectangle(bank,0,0,((T_R2D_FRAMEBUFFER*)the_frame_buffer)->width,
+        ((T_R2D_FRAMEBUFFER*)the_frame_buffer)->height);
+
+      #else
+		 gc->p_s_clipping_shape=r2d_new_rectangle(bank,0,0,((T_R2D_FRAMEBUFFER*)the_frame_buffer)->height,((T_R2D_FRAMEBUFFER*)the_frame_buffer)->width);
+
+      #endif
+
+	  if (gc->p_s_clipping_shape==NULL)
+	  {
+		  r2d_release_framebuffer(the_frame_buffer);
+          R2D_FREE(gc);
+          gc=NULL;
+		  return(NULL);
+	  }
+
+	  gc->p_clipping_shape=r2d_new_rectangle(bank,0,0,((T_R2D_FRAMEBUFFER*)the_frame_buffer)->width,
+      ((T_R2D_FRAMEBUFFER*)the_frame_buffer)->height);
+
+      
+      gc->drawing_mode=R2D_COPY_MODE;
+	  
+	  gc->drawing_op=r2d_get_drawing_op(gc,R2D_COPY_MODE);
+
+      gc->org_x=gc->org_y=0;
+	  gc->s_org_x=gc->s_org_y=0;
+
+      gc->pen_size=1;
+      gc->dash=0;
+      gc->text_face=R2D_SYSTEM;
+      gc->text_style=R2D_PLAIN;
+      gc->internal_text_style=R2D_PLAIN;
+      gc->text_size=R2D_SMALL;
+	  gc->script_mode=R2D_LEFT_TO_RIGHT;
+      gc->font_frame_buffer=r2d_g_default_font_framebuffer;
+      gc->font_metrics=r2d_g_default_font_metrics;
+	  gc->font_table=r2d_g_default_font_table;
+	  if (gc->p_clipping_shape)
+	  {
+        r2d_retain(r2d_g_default_font_framebuffer);
+      
+        // refcount set to 1 for creation
+        gc->refcount=1;
+        R2D_MALLOC(bank,UINT32,sizeof(UINT32)*4,gc->p_foreground_dithered_cache);
+        if (gc->p_foreground_dithered_cache)
+		{
+            R2D_MALLOC(bank,UINT32,sizeof(UINT32)*4,gc->p_background_dithered_cache);
+              if (gc->p_background_dithered_cache)
+			  {
+				if (((T_R2D_FRAMEBUFFER*)the_frame_buffer)->kind==R2D_FULL_KIND)
+				{
+				  r2d_convert_foreground_color_color(gc,gc->foreground_color);
+                  r2d_convert_background_color_color(gc,gc->background_color);
+				}
+				else
+				{
+                  r2d_convert_foreground_color(gc,gc->foreground_color);
+                  r2d_convert_background_color(gc,gc->background_color);
+				}
+			  } else // cannot create backround dithered matrix
+			  {
+                r2d_release_shape(gc->p_clipping_shape);
+				r2d_release_shape(gc->p_s_clipping_shape);
+                r2d_release_framebuffer(gc->p_frame_buffer);
+			    r2d_release_framebuffer(r2d_g_default_font_framebuffer);
+                R2D_FREE(((T_R2D_GC *)gc)->p_foreground_dithered_cache);
+				R2D_FREE(gc);
+                gc=NULL;
+			  }
+		} else // Cannot create foreground dithered matrix
+		{
+          r2d_release_shape(gc->p_clipping_shape);
+		  r2d_release_shape(gc->p_s_clipping_shape);
+          r2d_release_framebuffer(gc->p_frame_buffer);
+		  r2d_release_framebuffer(r2d_g_default_font_framebuffer);
+		  R2D_FREE(gc);
+          gc=NULL;
+		}
+	  } else // Cannot create clipping shape
+	  {
+		  R2D_FREE(gc);
+		  gc=NULL;
+	  }
+   }
+   else 
+   gc=NULL;
+   return(gc);
+}
+
+T_R2D_GC_PTR r2d_new_lcd_context(T_RVF_MB_ID bank)
+{
+//   T_R2D_GC *gc;
+   T_R2D_GC_PTR result;
+
+   result=r2d_new_context(bank,r2d_g_framebuffer);
+   r2d_release_framebuffer(r2d_g_framebuffer);
+   return (result);
+}
+
+void r2d_release_context(T_R2D_GC_PTR gc)
+{
+    if (gc)
+    {
+      if (R2D_REFCOUNT(gc)==1) 
+      {
+		  // framebuffer release since it is no more used
+        r2d_release_framebuffer(((T_R2D_GC *)gc)->font_frame_buffer);
+        r2d_release_shape(((T_R2D_GC *)gc)->p_clipping_shape);
+	    r2d_release_shape(((T_R2D_GC *)gc)->p_s_clipping_shape);
+        r2d_release_framebuffer(((T_R2D_GC *)gc)->p_frame_buffer);
+        R2D_FREE((((T_R2D_GC *)gc)->p_foreground_dithered_cache));
+        R2D_FREE((((T_R2D_GC *)gc)->p_background_dithered_cache));
+      }
+    }
+
+    // Autodelete of the gc if required
+    r2d_release(gc);
+}
+
+T_R2D_GC_PTR r2d_new_picture_context(T_RVF_MB_ID bank,const UINT32 *the_picture,T_R2D_FRAMEBUFFER_KIND kind)
+{
+   UINT32 h,v,length;
+   UINT32 *p;
+   T_R2D_FRAMEBUFFER *buf;
+   T_R2D_GC_PTR result;
+   T_RVF_RET err;
+
+   p=((UINT32*)the_picture)+3;
+
+   h=the_picture[0];
+   v=the_picture[1];
+   length=the_picture[2];
+
+   rvf_get_buf(bank, sizeof(T_R2D_FRAMEBUFFER), 
+   (T_RVF_BUFFER**)&buf);
+
+   if (buf != NULL)
+   {
+
+	   
+			
+           
+			
+			
+   buf->p_memory_words=p; 
+   buf->refcount=-1; // Emulate infinite refcount
+   buf->kind=kind; 
+   buf->p_frame_buffer_end=p+length;
+   buf->width=h; 
+   buf->height=v; 
+
+   R2D_MALLOC(bank,T_RVF_MUTEX,sizeof(T_RVF_MUTEX),buf->mutex);
+   if (buf->mutex!=NULL)
+   {
+		err=rvf_initialize_mutex(buf->mutex);
+		if (err!=RVF_OK)
+		{
+				 R2D_FREE(buf->mutex);
+				 R2D_FREE(buf);
+				 buf=NULL;
+				 return(buf);
+		}
+	}
+	else
+	{
+		R2D_FREE(buf);
+		buf=NULL;
+		return(buf);
+	}
+
+   result=r2d_new_context(bank,(T_R2D_FRAMEBUFFER_PTR)buf);
+   r2d_release_framebuffer(buf);
+
+   return(result);
+   } else return(NULL);
+}
+
+void r2d_release_picture_context(T_R2D_GC_PTR gc)
+{
+    T_R2D_FRAMEBUFFER *fb_p;
+
+    if (gc)
+    {
+        if (R2D_REFCOUNT(gc) == 1)
+        {
+            r2d_release_context(gc);
+            fb_p = (T_R2D_FRAMEBUFFER *) ((T_R2D_GC *) gc)->p_frame_buffer;
+            rvf_delete_mutex(fb_p->mutex);
+            R2D_FREE(fb_p->mutex);
+            R2D_FREE(fb_p);
+        }
+
+        r2d_release(gc);
+    }
+}
+
+UINT16          r2d_get_width(T_R2D_GC_PTR gc)
+{
+  return(((T_R2D_FRAMEBUFFER*)((T_R2D_GC *)gc)->p_frame_buffer)->width);
+}
+
+UINT16          r2d_get_height(T_R2D_GC_PTR gc)
+{
+  return(((T_R2D_FRAMEBUFFER*)((T_R2D_GC *)gc)->p_frame_buffer)->height);
+}
+
+void           r2d_erase(T_R2D_GC_PTR gc)
+{
+//   UINT32 *p;
+   INT16 x,y;
+   T_R2D_ARGB_COLOR col;
+
+   r2d_get_context_origin(gc,&x,&y);
+
+   col=r2d_get_background_color(gc);
+   r2d_set_background_color(gc,r2d_get_standard_argb_color(R2D_WHITE));
+
+   
+   r2d_fill_rectangle(gc,x-1,y-1,x+r2d_get_width(gc),y+r2d_get_height(gc));
+   r2d_set_background_color(gc,col);
+
+   if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->kind==0)
+      r2d_update_region((T_R2D_GC*)gc,x,y,x+r2d_get_width(gc)-1,y+r2d_get_height(gc)-1);
+   r2d_check_and_send_event(gc);
+}
+
+void           r2d_erase_with_background(T_R2D_GC_PTR gc,
+										 INT16 a,INT16 b,
+										 INT16 c,INT16 d)
+{
+   
+   r2d_fill_rectangle(gc,a-1,b-1,c+1,d+1);
+}
+
+T_R2D_DRAWING_MODE          r2d_get_drawing_mode(T_R2D_GC_PTR gc)
+{
+   return(((T_R2D_GC *)gc)->drawing_mode);
+}
+
+T_R2D_ERROR           r2d_set_drawing_mode(T_R2D_GC_PTR gc,T_R2D_DRAWING_MODE the_mode)
+{
+  if (the_mode!=((T_R2D_GC *)gc)->drawing_mode)
+  {
+	  // All modes must be supported in ASM mode
+
+	  if (r2d_get_drawing_op(gc,the_mode)!=NULL)
+
+	  {
+	      ((T_R2D_GC *)gc)->drawing_mode=the_mode;
+          ((T_R2D_GC *)gc)->drawing_op=r2d_get_drawing_op(gc,the_mode);
+
+		  return(R2D_OK);
+	  }
+
+	  else
+	   return(R2D_UNKNOWN_OP);
+
+  }
+  else
+	  return(R2D_OK);
+  
+  
+}
+
+UINT16          r2d_get_pen_size(T_R2D_GC_PTR gc)
+{
+   return(((T_R2D_GC *)gc)->pen_size);
+}
+
+void           r2d_set_pen_size(T_R2D_GC_PTR gc,UINT16 the_size)
+{
+   ((T_R2D_GC *)gc)->pen_size=the_size;
+}
+
+// H COORDINATES
+void           r2d_set_context_origin(T_R2D_GC_PTR gc,INT16 x,INT16 y)
+{
+  INT16 dx,dy;
+  T_R2D_RECTANGLE *ra,*rb;
+
+  dx=x-((T_R2D_GC *)gc)->org_x;
+  dy=y-((T_R2D_GC *)gc)->org_y;
+
+  ((T_R2D_GC *)gc)->org_x=x;
+  ((T_R2D_GC *)gc)->org_y=y;
+
+  r2d_translate_shape(((T_R2D_GC *)gc)->p_clipping_shape,dx,dy);
+
+
+  ra=r2d_new_rectangle(r2d_mb_id,0,0,0,0);
+  rb=r2d_s_level(gc,ra);
+
+  if (rb)
+  {
+    ((T_R2D_GC *)gc)->s_org_x=rb->ul_x;
+    ((T_R2D_GC *)gc)->s_org_y=rb->ul_y;
+  }
+
+  r2d_release_shape(ra);
+  r2d_release_shape(rb);
+  
+
+ // No translation of p_s_clipping_shape which is in S coordinates.
+  // Only H coordinates are impacted by the context origin
+}
+
+// H COORDINATES
+void           r2d_get_context_origin(T_R2D_GC_PTR gc,INT16 *x,INT16 *y)
+{
+  *x=((T_R2D_GC *)gc)->org_x;
+  *y=((T_R2D_GC *)gc)->org_y;
+}
+
+// H COORDINATES
+void           r2d_local_to_global(T_R2D_GC_PTR gc,INT16 *x,INT16 *y)
+{
+  *x-=((T_R2D_GC *)gc)->org_x;
+  *y-=((T_R2D_GC *)gc)->org_y;
+}
+
+// H coordinates
+void           r2d_get_pen_pos(T_R2D_GC_PTR gc,INT16 *x,INT16 *y)
+{
+  *x=((T_R2D_GC *)gc)->pen_x;
+  *y=((T_R2D_GC *)gc)->pen_y;
+}
+
+BOOLEAN        r2d_dash_enabled(T_R2D_GC_PTR gc)
+{
+   return(((T_R2D_GC *)gc)->dash);
+}
+
+void           r2d_set_dash_state(T_R2D_GC_PTR gc,BOOLEAN enabled)
+{
+   ((T_R2D_GC *)gc)->dash=enabled;
+}
+
+static void r2d_change_font(T_R2D_GC_PTR gc,T_R2D_FONT_DESCRIPTION *desc)
+{
+   r2d_retain(desc->framebuffer);
+   r2d_release_framebuffer(((T_R2D_GC *)gc)->font_frame_buffer);
+   ((T_R2D_GC *)gc)->font_frame_buffer=desc->framebuffer;
+
+   ((T_R2D_GC *)gc)->font_metrics=desc->metrics;
+   ((T_R2D_GC *)gc)->font_table=desc->table;
+}
+
+static T_R2D_FONT_DESCRIPTION *r2d_get_desc(T_R2D_GC_PTR gc)
+{
+    INT16 f;
+	INT16 s;
+	INT16 t;
+	T_R2D_FONT_DESCRIPTION *p;
+	INT16 temp;
+
+	p=r2d_g_font_configuration;
+
+	f=r2d_get_text_face(gc)-1;
+    s=r2d_get_text_size(gc)-1;
+    t=r2d_get_text_style(gc) & (R2D_BOLD | R2D_ITALIC);
+
+	t=t-(t>>1);
+
+	temp=(f<<2) + (f<<3) + (s<<2) + t;
+	p=r2d_g_font_configuration+(temp);
+
+	return(p);
+}
+
+void           r2d_set_text_face(T_R2D_GC_PTR gc,T_R2D_TEXT_FACE the_face)
+{
+  ((T_R2D_GC *)gc)->text_face=the_face;
+
+  r2d_change_font(gc,r2d_get_desc(gc));
+
+}
+
+T_R2D_TEXT_FACE r2d_get_text_face(T_R2D_GC_PTR gc)
+{
+  return(((T_R2D_GC *)gc)->text_face);
+}
+
+
+void           r2d_set_text_style(T_R2D_GC_PTR gc,T_R2D_TEXT_STYLE the_style)
+{
+  T_R2D_FONT_DESCRIPTION *desc;
+
+  ((T_R2D_GC *)gc)->text_style=the_style;
+
+  // Clear italic bit;
+  the_style=the_style & (~R2D_ITALIC);
+  
+  // Clear bold bit if desc for Bold is not NULL
+  desc=r2d_get_desc(gc);
+  if (!desc->compute_bold)
+     the_style=the_style & (~R2D_BOLD);
+  
+  
+  ((T_R2D_GC *)gc)->internal_text_style=the_style;
+
+  r2d_change_font(gc,r2d_get_desc(gc));
+  
+}
+
+T_R2D_TEXT_STYLE r2d_get_text_style(T_R2D_GC_PTR gc)
+{
+ return(((T_R2D_GC *)gc)->text_style);
+}
+
+
+void           r2d_set_text_size(T_R2D_GC_PTR gc,T_R2D_TEXT_SIZE the_size)
+{
+  ((T_R2D_GC *)gc)->text_size=the_size;
+  r2d_change_font(gc,r2d_get_desc(gc));
+}
+
+T_R2D_TEXT_SIZE r2d_get_text_size(T_R2D_GC_PTR gc)
+{
+  return(((T_R2D_GC *)gc)->text_size);
+}
+
+T_R2D_ERROR r2d_context_lock(T_R2D_GC_PTR gc)
+{
+	T_RVF_RET err;
+	err=RVF_OK;
+	err=rvf_lock_mutex(((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->mutex);
+	if (err!=RVF_OK)
+		return(R2D_LOCK_ERROR);
+	else
+	    return(R2D_OK);
+}
+
+T_R2D_ERROR r2d_context_unlock(T_R2D_GC_PTR gc)
+{
+	T_RVF_RET err;
+	err=RVF_OK;
+	err=rvf_unlock_mutex(((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->mutex);
+	if (err!=RVF_OK)
+		return(R2D_UNLOCK_ERROR);
+	else
+	    return(R2D_OK);
+}
+
+// Return the a pointer to the memory area containing
+// pixels for the graphic context gc
+//
+UINT32*        r2d_get_pixmap(T_R2D_GC_PTR gc)
+{
+	T_R2D_GC *src;
+	T_R2D_FRAMEBUFFER *src_f;
+	UINT32 *ps;
+
+    src=(T_R2D_GC*)gc;
+	src_f=(T_R2D_FRAMEBUFFER*)(src->p_frame_buffer);
+    ps=src_f->p_memory_words;
+
+	return(ps);
+}
+
+// Force the flushing of LCD framebuffer to display its
+// content 
+void r2d_flush(void)
+{
+	r2d_update_region(NULL,0,
+		  0,
+		  R2D_WIDTH-1,
+		  R2D_HEIGHT-1);
+    r2d_check_and_send_event(r2d_g_lcd_gc);
+}
+
+// xreddymn Apr-06-2005 MMI-SPR-29655
+// Force the flushing of a portion of LCD framebuffer to display its
+// content 
+void r2d_flush_region(INT16 ul_x,INT16 ul_y,INT16 br_x,INT16 br_y)
+{
+	r2d_update_region(NULL, ul_x,
+		  ul_y,
+		  br_x,
+		  br_y);
+    r2d_check_and_send_event(r2d_g_lcd_gc);
+}
+
+void    r2d_disable_refresh(void) 
+{
+	rvf_lock_mutex(r2d_g_global_mutex);
+	r2d_g_refresh_disabled++;
+	//rvf_send_trace("R2D : disable refresh\n",sizeof("R2D : disable refresh\n"), NULL_PARAM, 
+	//			   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+	rvf_unlock_mutex(r2d_g_global_mutex);
+}
+
+void r2d_enable_refresh(void) 
+{
+	//rvf_send_trace("R2D : enable refresh\n",sizeof("R2D : enable refresh\n"), NULL_PARAM, 
+	//			   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+	rvf_lock_mutex(r2d_g_global_mutex);
+	r2d_g_refresh_disabled--;
+    if (r2d_g_refresh_disabled==0)
+	{
+		//rvf_send_trace("R2D : flush\n",sizeof("R2D : flush\n"), NULL_PARAM, 
+				   //RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+	  //rvf_send_trace("Enable refresh and flush",sizeof("Enable refresh and flush")-1, NULL_PARAM, 
+	  //		   RV_TRACE_LEVEL_DEBUG_HIGH, R2D_USE_ID );
+		// The flush will not raise a context commutation to R2D since R2D may
+		// be sleeping waiting for the current refresh perio.
+		// So, the event will not be processed at once. While R2D is sleeping,
+		// another task may call r2d_disable_refresh and thus
+		// when R2D is awaken, it will not process the event which is waiting.
+		// So, finally, the display will not be refreshed.
+		// To avoid this problem which occurs for a quick succession of
+		// r2d_disable_refresh
+		// r2d_enable_refresh
+		// one should try to call them with a rate which is lower than the refresh rate
+        rvf_unlock_mutex(r2d_g_global_mutex);
+		//r2d_flush();
+		
+        r2d_check_and_send_event(r2d_g_lcd_gc);
+	}
+	else
+	rvf_unlock_mutex(r2d_g_global_mutex);
+}
+
+////////////////////////////////////////
+//
+// Colors
+//
+
+
+// Return ARGB foreground color for the gc
+T_R2D_ARGB_COLOR r2d_get_foreground_color(T_R2D_GC_PTR gc)
+{
+  return (((T_R2D_GC*)gc)->foreground_color);
+}
+
+// Set ARGB foreground color for the gc
+// (Alpha channel not yet supported)
+void r2d_set_foreground_color_with_argb(T_R2D_GC_PTR gc,UINT16 alpha,UINT16 red,UINT16 green,UINT16 blue)
+{
+ UINT32 color;
+ color=r2d_new_argb_color(alpha,red,green,blue);
+ if (color !=((T_R2D_GC*)gc)->foreground_color)
+ {
+   ((T_R2D_GC*)gc)->foreground_color=color;
+
+   if (((T_R2D_FRAMEBUFFER*)(((T_R2D_GC*)gc)->p_frame_buffer))->kind==R2D_FULL_KIND)
+r2d_convert_foreground_color_color(gc,color);
+   else
+   r2d_convert_foreground_color(gc,color);
+ }
+}
+
+// Set ARGB foreground color for the gc
+// (Alpha channel not yet supported)
+void r2d_set_foreground_color(T_R2D_GC_PTR gc,T_R2D_ARGB_COLOR color)
+{
+ if (color !=((T_R2D_GC*)gc)->foreground_color)
+ {
+   ((T_R2D_GC*)gc)->foreground_color=color;
+   if (((T_R2D_FRAMEBUFFER*)(((T_R2D_GC*)gc)->p_frame_buffer))->kind==R2D_FULL_KIND)
+  r2d_convert_foreground_color_color(gc,color);
+	   else
+   r2d_convert_foreground_color(gc,color);
+ }
+}
+
+// Return ARGB foreground color for the gc
+T_R2D_ARGB_COLOR r2d_get_background_color(T_R2D_GC_PTR gc)
+{
+  return (((T_R2D_GC*)gc)->background_color);
+}
+
+// Set ARGB foreground color for the gc
+// (Alpha channel not yet supported)
+void r2d_set_background_color_with_argb(T_R2D_GC_PTR gc,UINT16 alpha,UINT16 red,UINT16 green,UINT16 blue)
+{
+ UINT32 color;
+ color=r2d_new_argb_color(alpha,red,green,blue);
+ if (color !=((T_R2D_GC*)gc)->background_color)
+ {
+   ((T_R2D_GC*)gc)->background_color=color;
+   if (((T_R2D_FRAMEBUFFER*)(((T_R2D_GC*)gc)->p_frame_buffer))->kind==R2D_FULL_KIND)
+  r2d_convert_background_color_color(gc,color);
+	   else
+   r2d_convert_background_color(gc,color);
+ }
+}
+
+// Set ARGB foreground color for the gc
+// (Alpha channel not yet supported)
+void r2d_set_background_color(T_R2D_GC_PTR gc,T_R2D_ARGB_COLOR color)
+{
+ if (color !=((T_R2D_GC*)gc)->background_color)
+ {
+   ((T_R2D_GC*)gc)->background_color=color;
+   if (((T_R2D_FRAMEBUFFER*)(((T_R2D_GC*)gc)->p_frame_buffer))->kind==R2D_FULL_KIND)
+	   r2d_convert_background_color_color(gc,color);
+  else
+   r2d_convert_background_color(gc,color);
+ }
+}
+
+void   r2d_set_background_texture(T_R2D_GC_PTR gc,T_R2D_ANCHORED_TEXTURE_PTR texture)
+{
+	
+//    INT16 length;
+//	T_R2D_ANCHORED_TEXTURE *t;
+
+	// If a texture is already in place try to release it
+	if (((T_R2D_GC*)gc)->background_texture!=NULL)
+	{
+		r2d_release_anchored_texture(((T_R2D_GC*)gc)->background_texture);
+		((T_R2D_GC*)gc)->background_texture=NULL;
+	}
+	
+
+	((T_R2D_GC*)gc)->background_texture=texture;
+     
+
+   if (texture!=NULL)
+   {
+     r2d_retain(texture);
+   }
+
+  
+
+	
+}
+
+T_R2D_ANCHORED_TEXTURE_PTR r2d_get_background_texture(T_R2D_GC_PTR gc)
+{
+	return ((T_R2D_GC*)gc)->background_texture;
+}
+
+T_R2D_ARGB_COLOR         r2d_new_argb_color(UINT16 alpha,UINT16 red,UINT16 green,UINT16 blue)
+{
+   T_R2D_ARGB_COLOR col;
+   col=0;
+
+   col=col|(alpha&0xFF);
+   col=col<<8;
+   col=col|(red&0xFF);
+   col=col<<8;
+   col=col|(green&0xFF);
+   col=col<<8;
+   col=col|(blue&0xFF);
+  
+   return(col);
+}
+
+T_R2D_ARGB_COLOR         r2d_get_argb_color_at_point(T_R2D_GC_PTR gc,INT16 x,INT16 y)
+{
+  UINT32 result;
+  if (r2d_point_in_shape(((T_R2D_GC*)gc)->p_clipping_shape,x,y))
+  {
+	r2d_local_to_global(gc,&x,&y);
+	#ifdef R2D_MIRRORED_X
+	
+	  x=r2d_get_width(gc)-1-x;
+    #endif
+
+    #ifdef R2D_MIRRORED_Y
+	  y=r2d_get_height(gc)-1-y;
+    #endif
+	
+#if (R2D_REFRESH == R2D_VERTICAL)
+	result=IND_r2d_get_color_pixel_value(gc,x,y);
+#else
+	result=IND_r2d_get_color_pixel_value(gc,y,x);
+#endif
+    return(result);
+  } 
+  else
+  return(0);
+}
+
+T_R2D_ARGB_COLOR         r2d_get_standard_argb_color(T_R2D_COLOR_REF ref)
+{
+  switch(ref)
+  {
+     case R2D_RED: return(r2d_new_argb_color(0,255,0,0)); break;
+     case R2D_BLUE:return(r2d_new_argb_color(0,0,0,255)); break;
+     case R2D_GREEN:return(r2d_new_argb_color(0,0,255,0)); break;
+     case R2D_WHITE:return(r2d_new_argb_color(0,255,255,255)); break;
+     case R2D_BLACK:return(r2d_new_argb_color(0,0,0,0)); break;
+     case R2D_GRAY50:return(r2d_new_argb_color(0,127,127,12)); break;
+     default:return(r2d_new_argb_color(0,0,0,0)); break;
+  }
+}
+
+// h from 0 to 360 : 
+// s from 0 to 1 : 1 is 0x800000
+// v from 0 to 1
+// r,g,b from 0 to 1
+// Value in Q8
+void r2d_hsv_to_rgb( INT16 *r, INT16 *g, INT16 *b, INT32 h, INT32 s, INT32 v )
+{
+        int i;
+        INT32 f, p, q, t;
+
+		//s and v in Q8. So 255 is FF is 1
+		
+        if( s == 0 ) {
+                // achromatic (grey)
+                *r = *g = *b = (v>>13);
+                return;
+        }
+        h = h * (0x3000000 >> 8) ; // 360/256 / 60 ;Q23 ; sector 0 to 5
+        i = h & 0xFF800000;
+        f = h - i;                      // factorial part of h
+		i=i>>15;
+        f = f >> 15; // Q16
+        p = (v * ( 0x100 - s )) >> 8;
+        q = (v * ( 0x100 - ((s * f) >> 8) )) >> 8;
+        t = (v * ( 0x100 - ((s * ( 0x100 - f )) >> 8) )) >> 8;
+        switch( i>>8 ) {
+                case 0:
+                        *r = v;
+                        *g = t;
+                        *b = p;
+                        break;
+                case 1:
+                        *r = q;
+                        *g = v;
+                        *b = p;
+                        break;
+                case 2:
+                        *r = p;  
+                        *g = v;
+                        *b = t;
+                        break;
+                case 3:
+                        *r = p;
+                        *g = q;
+                        *b = v;
+                        break;
+                case 4:
+                        *r = t;
+                        *g = p;
+                        *b = v;
+                        break; 
+                default:                // case 5:
+                        *r = v;
+                        *g = p;
+                        *b = q;
+                        break; 
+        }
+}
+
+////////////////////////////////////////
+//
+// Shapes
+//
+
+// H COORDINATES ...
+void r2d_draw_shape(T_R2D_SHAPE_PTR self,T_R2D_GC_PTR gc)
+{
+	
+    if (self)
+	{
+		((T_R2D_SHAPE*)self)->p_r2d_class->draw(self,gc);
+	  
+	}
+}
+
+void r2d_fill_shape(T_R2D_SHAPE_PTR self,T_R2D_GC_PTR gc)
+{
+    if (self)
+	{
+      ((T_R2D_SHAPE*)self)->p_r2d_class->fill(self,gc);
+	  
+	}
+}
+
+void r2d_release_shape(T_R2D_SHAPE_PTR self)
+{
+
+
+   if (self)
+     ((T_R2D_SHAPE*)self)->p_r2d_class->release(self);
+}
+
+T_R2D_SHAPE_PTR r2d_clone_shape(T_RVF_MB_ID bank,T_R2D_SHAPE_PTR self)
+{
+   if (self)
+     return (((T_R2D_SHAPE*)self)->p_r2d_class->clone(bank,self));
+   else
+     return(NULL);
+}
+
+void r2d_translate_shape(T_R2D_SHAPE_PTR self,INT16 dx,INT16 dy)
+{
+   if (self)
+     ((T_R2D_SHAPE*)self)->p_r2d_class->translate(self,dx,dy);
+     
+} 
+
+BOOLEAN r2d_point_in_shape(T_R2D_SHAPE_PTR self,INT16 x,INT16 y)
+{
+
+  if (self)
+     return(((T_R2D_SHAPE*)self)->p_r2d_class->point_in_shape(self,x,y));
+  else
+      return(FALSE);
+
+}
+
+
+BOOLEAN r2d_point_in_rectangle_shape(T_R2D_SHAPE* self,INT16 x,INT16 y)
+{
+	T_R2D_RECTANGLE *r;
+
+	r=(T_R2D_RECTANGLE*)self;
+
+    return (((x>=r2d_get_xmin(self))
+    && (x<r2d_get_xmax(self))
+    && (y>=r2d_get_ymin(self))
+    && (y<r2d_get_ymax(self))));
+}
+
+BOOLEAN r2d_point_in_circle_shape(T_R2D_SHAPE* self,INT16 x,INT16 y)
+{
+	T_R2D_CIRCLE *c;
+	c=(T_R2D_CIRCLE*)self;
+
+	return((x-c->x)*(x-c->x)+(y-c->y)*(y-c->y)-c->r*c->r<0);
+}
+
+BOOLEAN r2d_point_in_ellipse_shape(T_R2D_SHAPE* self,INT16 x,INT16 y)
+{
+	T_R2D_ELLIPSE *c;
+	INT32 a,b,cx,cy,nx,ny;
+
+	c=(T_R2D_ELLIPSE*)self;
+	// Work on Q1
+	
+	// a is half width so no <<1 is used here
+	a=(c->br_x-c->ul_x);
+	b=(c->br_y-c->ul_y);
+
+	nx=x<<1;
+	ny=y<<1;
+
+	cx=((c->ul_x)<<1)+a;
+    cy=((c->ul_y)<<1)+b;
+
+    //b^2*x^2+a^2y^2-a^2b^2=0
+
+	return((b*b*(nx-cx)*(nx-cx)+a*a*(ny-cy)*(ny-cy)-a*a*b*b)<0);
+
+}
+
+BOOLEAN r2d_point_in_round_rectangle_shape(T_R2D_SHAPE* self,INT16 x,INT16 y)
+{
+  T_R2D_ROUND_RECTANGLE *r;
+  BOOLEAN result;
+
+  result=FALSE;
+  r=(T_R2D_ROUND_RECTANGLE *)self;
+
+  if ((x>=r->ul_x+r->h+1) && (x<r->br_x-r->h) && (y>=r->ul_y) && (y<r->br_y))
+	  result=TRUE;
+  else if ((x>=r->ul_x) && (x<r->br_x) && (y>=r->ul_y+r->v+1) && (y<r->br_y-r->v))
+	  result=TRUE;
+  else
+  {
+	  INT32 a,b,cx,cy,nx,ny;
+
+      if (x>=r->br_x-r->h)
+		  x=x-(r->br_x-r->ul_x)+((r->h)<<1);
+
+      if (y>=r->br_y-r->v)
+	      y=y-(r->br_y-r->ul_y)+((r->v)<<1);
+
+	  
+	  a=(r->h<<1);
+	  b=(r->v<<1);
+
+	  nx=x<<1;
+	  ny=y<<1;
+
+	  cx=((r->ul_x)<<1)+a;
+      cy=((r->ul_y)<<1)+b;
+
+    //b^2*x^2+a^2y^2-a^2b^2=0
+
+	 result=(b*b*(nx-cx)*(nx-cx)+a*a*(ny-cy)*(ny-cy)-a*a*b*b)<0;
+
+   
+  }
+  
+  return(result);
+  
+}
+
+BOOLEAN r2d_point_in_not_supported_shape(T_R2D_SHAPE* self,INT16 x,INT16 y)
+{
+    return FALSE;
+}
+
+void r2d_translate_rectangle_shape(T_R2D_SHAPE* self,INT16 dx,INT16 dy)
+{
+   T_R2D_RECTANGLE *r;
+   r=(T_R2D_RECTANGLE *)self;
+   r->ul_x+=dx;
+   r->ul_y+=dy;
+   r->br_x+=dx;
+   r->br_y+=dy;
+} 
+
+void r2d_translate_point_shape(T_R2D_SHAPE* self,INT16 dx,INT16 dy)
+{
+   T_R2D_CIRCLE *r;
+   r=(T_R2D_CIRCLE *)self;
+   r->x+=dx;
+   r->y+=dy;
+} 
+
+
+T_R2D_SHAPE_PTR r2d_new_rectangle(T_RVF_MB_ID bank,INT16 ul_x,INT16 ul_y,INT16 br_x,INT16 br_y)
+{
+  T_R2D_RECTANGLE *r;
+  T_R2D_ERROR err;
+  err=R2D_MALLOC(bank,T_R2D_RECTANGLE,sizeof(T_R2D_RECTANGLE),r);
+  if (r)
+  {
+  r->refcount=1;
+  r->ul_x=ul_x;
+  r->ul_y=ul_y;
+  r->br_x=br_x;
+  r->br_y=br_y;
+  r->p_r2d_class=&r2d_class_rectangle;
+  } else r=NULL;
+  return(r);
+}
+
+void r2d_release_rectangle(T_R2D_SHAPE* self)
+{
+   T_R2D_RECTANGLE *r;
+   r=(T_R2D_RECTANGLE *)self;
+
+   r2d_release(r);
+}
+
+T_R2D_SHAPE_PTR r2d_clone_rectangle_shape(T_RVF_MB_ID bank,T_R2D_SHAPE* self)
+{
+   T_R2D_RECTANGLE *r;
+   r=(T_R2D_RECTANGLE *)self;
+
+   return(r2d_new_rectangle(bank,r->ul_x,r->ul_y,r->br_x,r->br_y));
+}
+
+void r2d_draw_rectangle_shape(T_R2D_SHAPE* self,T_R2D_GC_PTR gc)
+{
+   T_R2D_RECTANGLE *r;
+   r=(T_R2D_RECTANGLE *)self;
+   r2d_draw_rectangle(gc,r->ul_x,r->ul_y,r->br_x,r->br_y);
+}
+
+void r2d_fill_rectangle_shape(T_R2D_SHAPE* self,T_R2D_GC_PTR gc)
+{
+   T_R2D_RECTANGLE *r;
+   r=(T_R2D_RECTANGLE *)self;
+   r2d_fill_rectangle(gc,r->ul_x,r->ul_y,r->br_x,r->br_y);
+}
+
+
+void r2d_release_circle(T_R2D_SHAPE* self)
+{
+   T_R2D_CIRCLE *r;
+   r=(T_R2D_CIRCLE *)self;
+
+   r2d_release(r);
+
+   #if (R2D_DEBUG == R2D_ON)
+   r=NULL;
+   #endif
+}
+
+T_R2D_SHAPE_PTR r2d_clone_circle_shape(T_RVF_MB_ID bank,T_R2D_SHAPE* self)
+{
+   T_R2D_CIRCLE *r;
+   r=(T_R2D_CIRCLE *)self;
+
+   return(r2d_new_circle(bank,r->x,r->y,r->r));
+}
+
+void r2d_draw_circle_shape(T_R2D_SHAPE* self,T_R2D_GC_PTR gc)
+{
+   T_R2D_CIRCLE *r;
+   r=(T_R2D_CIRCLE *)self;
+   r2d_draw_circle(gc,r->x,r->y,r->r);
+}
+
+void r2d_fill_circle_shape(T_R2D_SHAPE* self,T_R2D_GC_PTR gc)
+{
+   T_R2D_CIRCLE *r;
+   r=(T_R2D_CIRCLE *)self;
+   r2d_fill_circle(gc,r->x,r->y,r->r);
+}
+
+T_R2D_SHAPE_PTR r2d_new_circle(T_RVF_MB_ID bank,INT16 x,INT16 y,INT16 r)
+{
+  T_R2D_CIRCLE *c;
+  R2D_MALLOC(bank,T_R2D_CIRCLE,sizeof(T_R2D_CIRCLE),c);
+  if (c)
+  {
+  c->refcount=1;
+  c->x=x;
+  c->y=y;
+  c->r=r;
+  c->p_r2d_class=&r2d_class_circle;
+  } else c=NULL;
+  return(c);
+}
+
+void r2d_release_ellipse(T_R2D_SHAPE* self)
+{
+   T_R2D_ELLIPSE *r;
+   r=(T_R2D_ELLIPSE *)self;
+
+   r2d_release(r);
+#if (R2D_DEBUG == R2D_ON)
+   r=NULL;
+#endif
+}
+
+void r2d_draw_ellipse_shape(T_R2D_SHAPE* self,T_R2D_GC_PTR gc)
+{
+   T_R2D_ELLIPSE *r;
+   r=(T_R2D_ELLIPSE *)self;
+   r2d_draw_ellipse(gc,r->ul_x,r->ul_y,r->br_x,r->br_y);
+}
+
+void r2d_fill_ellipse_shape(T_R2D_SHAPE* self,T_R2D_GC_PTR gc)
+{
+   T_R2D_ELLIPSE *r;
+   r=(T_R2D_ELLIPSE *)self;
+   r2d_fill_ellipse(gc,r->ul_x,r->ul_y,r->br_x,r->br_y);
+}
+
+T_R2D_SHAPE_PTR r2d_new_ellipse(T_RVF_MB_ID bank,INT16 ul_x,INT16 ul_y,INT16 br_x,INT16 br_y)
+{
+  T_R2D_ELLIPSE *r;
+  R2D_MALLOC(bank,T_R2D_ELLIPSE,sizeof(T_R2D_ELLIPSE),r);
+  if (r)
+  {
+  r->refcount=1;
+  r->ul_x=ul_x;
+  r->ul_y=ul_y;
+  r->br_x=br_x;
+  r->br_y=br_y;
+  r->p_r2d_class=&r2d_class_ellipse;
+  } else r=NULL;
+  return(r);
+}
+
+void r2d_release_round_rectangle(T_R2D_SHAPE* self)
+{
+   T_R2D_ROUND_RECTANGLE *r;
+   r=(T_R2D_ROUND_RECTANGLE *)self;
+
+   r2d_release(r);
+   #if (R2D_DEBUG == R2D_ON)
+   r=NULL;
+#endif
+}
+
+void r2d_draw_round_rectangle_shape(T_R2D_SHAPE* self,T_R2D_GC_PTR gc)
+{
+   T_R2D_ROUND_RECTANGLE *r;
+   r=(T_R2D_ROUND_RECTANGLE *)self;
+   r2d_draw_round_rectangle(gc,r->ul_x,r->ul_y,r->br_x,r->br_y,r->h,r->v);
+}
+
+void r2d_fill_round_rectangle_shape(T_R2D_SHAPE* self,T_R2D_GC_PTR gc)
+{
+   T_R2D_ROUND_RECTANGLE *r;
+   r=(T_R2D_ROUND_RECTANGLE *)self;
+   r2d_fill_round_rectangle(gc,r->ul_x,r->ul_y,r->br_x,r->br_y,r->h,r->v);
+}
+
+T_R2D_SHAPE_PTR r2d_clone_round_rectangle_shape(T_RVF_MB_ID bank,T_R2D_SHAPE* self)
+{
+   T_R2D_ROUND_RECTANGLE *r;
+   r=(T_R2D_ROUND_RECTANGLE *)self;
+
+   return(r2d_new_round_rectangle(bank,r->ul_x,r->ul_y,r->br_x,r->br_y,r->h,r->v));
+}
+
+T_R2D_SHAPE_PTR r2d_new_round_rectangle(T_RVF_MB_ID bank,INT16 ul_x,INT16 ul_y,INT16 br_x,INT16 br_y,INT16 h,INT16 v)
+{
+  T_R2D_ROUND_RECTANGLE *r;
+  R2D_MALLOC(bank,T_R2D_ROUND_RECTANGLE,sizeof(T_R2D_ROUND_RECTANGLE),r);
+  if (r)
+  {
+  r->refcount=1;
+  r->ul_x=ul_x;
+  r->ul_y=ul_y;
+  r->br_x=br_x;
+  r->br_y=br_y;
+  r->h=h;
+  r->v=v;
+  r->p_r2d_class=&r2d_class_round_rectangle;
+  } else r=NULL;
+  return(r);
+}
+
+T_R2D_SHAPE_PTR r2d_clone_arc_shape(T_RVF_MB_ID bank,T_R2D_SHAPE* self)
+{
+   T_R2D_ARC *r;
+   r=(T_R2D_ARC *)self;
+
+   return(r2d_new_arc(bank,r->start_angle,r->stop_angle,r->ul_x,r->ul_y,r->br_x,r->br_y));
+}
+
+T_R2D_SHAPE_PTR r2d_new_arc(T_RVF_MB_ID bank,INT16 start_angle, INT16 stop_angle,INT16 ul_x,INT16 ul_y,
+                                         INT16 br_x,INT16 br_y)
+{
+  T_R2D_ARC *r;
+  R2D_MALLOC(bank,T_R2D_ARC,sizeof(T_R2D_ARC),r);
+  if (r)
+  {
+  r->refcount=1;
+  r->ul_x=ul_x;
+  r->ul_y=ul_y;
+  r->br_x=br_x;
+  r->br_y=br_y;
+  r->start_angle=start_angle;
+  r->stop_angle=stop_angle;
+  r->p_r2d_class=&r2d_class_arc;
+  } else r=NULL;
+  return(r);
+}
+
+void r2d_release_arc(T_R2D_SHAPE *self)
+{
+  T_R2D_ARC *r;
+  r=(T_R2D_ARC *)self;
+
+  r2d_release(r);
+}
+
+void r2d_draw_arc_shape(T_R2D_SHAPE *self,T_R2D_GC_PTR gc)
+{
+  T_R2D_ARC *r;
+  r=(T_R2D_ARC *)self;
+
+	r2d_draw_arc(gc,r->start_angle,r->stop_angle,
+                  r->ul_x,r->ul_y,r->br_x,r->br_y);
+}
+
+void r2d_fill_arc_shape(T_R2D_SHAPE *self,T_R2D_GC_PTR gc)
+{
+ T_R2D_ARC *r;
+  r=(T_R2D_ARC *)self;
+
+	r2d_fill_arc(gc,r->start_angle,r->stop_angle,
+                  r->ul_x,r->ul_y,r->br_x,r->br_y);
+}
+
+T_R2D_SHAPE_PTR r2d_new_rectangle_intersection(T_RVF_MB_ID bank,T_R2D_SHAPE_PTR a,T_R2D_SHAPE_PTR b)
+{
+if ( a && b && (((T_R2D_SHAPE*)a)->p_r2d_class==&r2d_class_rectangle)
+&& (((T_R2D_SHAPE*)b)->p_r2d_class==&r2d_class_rectangle))
+{
+	T_R2D_RECT *ra;
+	T_R2D_RECT *rb;
+	INT16 xa,ya,xb,yb;
+
+	ra=(T_R2D_RECT *)a;
+	rb=(T_R2D_RECT *)b;
+
+    if (ra->ul_x>rb->ul_x)
+		xa=ra->ul_x;
+	else
+		xa=rb->ul_x;
+
+    if (ra->ul_y>rb->ul_y)
+		ya=ra->ul_y;
+	else
+		ya=rb->ul_y;
+
+	if (ra->br_x<rb->br_x)
+		xb=ra->br_x;
+	else
+		xb=rb->br_x;
+
+    if (ra->br_y<rb->br_y)
+		yb=ra->br_y;
+	else
+		yb=rb->br_y;
+
+	if ((xa<=xb) && (ya<=yb))
+		return(r2d_new_rectangle(bank,xa,ya,xb,yb));
+	else
+		return(R2D_EMPTY_RECT);
+
+}
+else return(NULL);
+}
+
+T_R2D_SHAPE_PTR r2d_new_rectangle_union(T_RVF_MB_ID bank,T_R2D_SHAPE_PTR a,T_R2D_SHAPE_PTR b)
+{
+	if ( a && b && (((T_R2D_SHAPE*)a)->p_r2d_class==&r2d_class_rectangle)
+&& (((T_R2D_SHAPE*)b)->p_r2d_class==&r2d_class_rectangle))
+{
+	T_R2D_RECTANGLE *ra;
+	T_R2D_RECTANGLE *rb;
+	INT16 xa,ya,xb,yb;
+
+	ra=(T_R2D_RECTANGLE *)a;
+	rb=(T_R2D_RECTANGLE *)b;
+
+    if (ra->ul_x<rb->ul_x)
+		xa=ra->ul_x;
+	else
+		xa=rb->ul_x;
+
+    if (ra->ul_y<rb->ul_y)
+		ya=ra->ul_y;
+	else
+		ya=rb->ul_y;
+
+	if (ra->br_x>rb->br_x)
+		xb=ra->br_x;
+	else
+		xb=rb->br_x;
+
+    if (ra->br_y>rb->br_y)
+		yb=ra->br_y;
+	else
+		yb=rb->br_y;
+
+	if ((xa<=xb) && (ya<=yb))
+		return(r2d_new_rectangle(bank,xa,ya,xb,yb));
+	else
+		return(R2D_EMPTY_RECT);
+
+}
+else if (a) 
+return(r2d_clone_shape(bank,a));
+else if (b)
+return(r2d_clone_shape(bank,b));
+else
+return(NULL);
+}
+
+T_R2D_SHAPE_PTR r2d_new_text(T_RVF_MB_ID bank,INT16 x, INT16 y,T_R2D_UTF16 *the_text)
+{
+  T_R2D_TEXT *r;
+  R2D_MALLOC(bank,T_R2D_TEXT,sizeof(T_R2D_TEXT),r);
+  if (r)
+  {
+  r->refcount=1;
+  r->x=x;
+  r->y=y;
+  r->p_text=the_text;
+  r->p_r2d_class=&r2d_class_text;
+  } else r=NULL;
+  return(r);
+}
+
+T_R2D_SHAPE_PTR r2d_clone_text_shape(T_RVF_MB_ID bank,T_R2D_SHAPE* self)
+{
+   T_R2D_TEXT *r;
+   r=(T_R2D_TEXT *)self;
+
+   return(r2d_new_text(bank,r->x,r->y,NULL));
+}
+
+
+void r2d_release_text(T_R2D_SHAPE *self)
+{
+  T_R2D_TEXT *r;
+  r=(T_R2D_TEXT *)self;
+
+  r2d_release(r);
+}
+
+void r2d_draw_text_shape(T_R2D_SHAPE *self,T_R2D_GC_PTR gc)
+{
+	T_R2D_TEXT *r;
+	r=(T_R2D_TEXT *)self;
+	r2d_draw_text(gc,r->x,r->y,r->p_text);
+}
+
+void r2d_fill_text_shape(T_R2D_SHAPE *self,T_R2D_GC_PTR gc)
+{
+    // A text has no interior and thus cannot be filled.
+}
+
+
+
+
+////////////////////////////////////////
+//
+// Clipping functions
+//
+
+// Conversion is done such that line y = ymax and x=xmax MUST NOT
+// be drawn. So, they do not belong to the cliiping region
+// So, behavior of s_clipping_shape is using drawing convention
+// and not the clipping convention
+static T_R2D_SHAPE_PTR r2d_s_level(T_R2D_GC_PTR gc,T_R2D_SHAPE_PTR self)
+{
+	T_R2D_SHAPE_PTR result;
+	T_R2D_RECTANGLE *r;
+
+	
+	r=(T_R2D_RECTANGLE*)self;
+
+	if ((r) && (gc))
+	{
+
+	result=r2d_clone_shape(r2d_mb_id,self);
+	r=(T_R2D_RECTANGLE*)result;
+
+	r2d_local_to_global(gc,&(r->ul_x),&(r->ul_y));
+	r2d_local_to_global(gc,&(r->br_x),&(r->br_y));
+
+	//r2d_mirror_clip_rectangle(gc,result);
+	r2d_mirror_rectangle(gc,result);
+	r=(T_R2D_RECTANGLE*)result;
+
+#if (R2D_REFRESH != R2D_VERTICAL)
+	r2d_diagonal_mirror(result);
+	r=(T_R2D_RECTANGLE*)result;
+#endif
+	
+	return(result);
+	}
+	else
+		return(NULL);
+}
+
+void r2d_set_clipping_shape(T_R2D_SHAPE_PTR self,T_R2D_GC_PTR gc)
+{
+
+   if ((self) && (((T_R2D_SHAPE*)self)->p_r2d_class==&r2d_class_rectangle))
+   {
+       T_R2D_RECT *r;
+	   INT16 orgx,orgy;
+
+       r=(T_R2D_RECT *)self;
+
+	   r2d_get_context_origin(gc,&orgx,&orgy);
+
+	   if (r->ul_x-orgx<0)
+		   r->ul_x=orgx;
+
+	   if (r->ul_y-orgy<0)
+		   r->ul_y=orgy;
+
+	   if (r->br_x-orgx>=r2d_get_width(gc))
+		   r->br_x=orgx+r2d_get_width(gc);
+
+	   if (r->br_y-orgy>=r2d_get_height(gc))
+		  r->br_y=orgy+r2d_get_height(gc);
+
+		  
+       {
+       r2d_release_shape(((T_R2D_GC *)gc)->p_clipping_shape);
+       ((T_R2D_GC *)gc)->p_clipping_shape=self;
+       r2d_retain(self);
+
+	   r2d_release_shape(((T_R2D_GC *)gc)->p_s_clipping_shape);
+       ((T_R2D_GC *)gc)->p_s_clipping_shape=r2d_s_level(gc,self);
+	   }
+   }
+}
+
+T_R2D_SHAPE_PTR r2d_get_clipping_shape(T_R2D_GC_PTR gc)
+{
+   return(((T_R2D_GC *)gc)->p_clipping_shape);
+}
+
+T_R2D_ERROR r2d_restore_standard_clipping_shape(T_R2D_GC_PTR gc)
+{
+    INT16 x,y;
+	T_R2D_SHAPE_PTR s,sb;
+    r2d_get_context_origin(gc,&x,&y);
+	s=((T_R2D_GC *)gc)->p_clipping_shape;
+	sb=((T_R2D_GC *)gc)->p_s_clipping_shape;
+    
+    ((T_R2D_GC *)gc)->p_clipping_shape=r2d_new_rectangle(r2d_mb_id,x,y,x+r2d_get_width(gc),y+r2d_get_height(gc));
+    
+// No mirroring when shape is the whole LCD
+// Force use of global coordinates for s_ frame
+#if (R2D_REFRESH == R2D_VERTICAL)
+	((T_R2D_GC *)gc)->p_s_clipping_shape=
+		r2d_new_rectangle(r2d_mb_id,0,0,r2d_get_width(gc),r2d_get_height(gc));
+#else
+	((T_R2D_GC *)gc)->p_s_clipping_shape=
+		r2d_new_rectangle(r2d_mb_id,0,0,r2d_get_height(gc),r2d_get_width(gc));
+#endif
+
+	 
+	if ((((T_R2D_GC *)gc)->p_clipping_shape==NULL) 
+		|| (((T_R2D_GC *)gc)->p_s_clipping_shape==NULL))
+	{
+		if (((T_R2D_GC *)gc)->p_clipping_shape!=NULL)
+          r2d_release_shape(((T_R2D_GC *)gc)->p_clipping_shape);
+		if (((T_R2D_GC *)gc)->p_s_clipping_shape!=NULL)
+          r2d_release_shape(((T_R2D_GC *)gc)->p_s_clipping_shape);
+        ((T_R2D_GC *)gc)->p_clipping_shape=s;
+		((T_R2D_GC *)gc)->p_s_clipping_shape=sb;
+		return(R2D_MEMORY_ERR);
+	}
+	else
+	{
+	    r2d_release_shape(s);
+		r2d_release_shape(sb);
+		return(R2D_OK);
+	}
+}
+
+
+////////////////////////////////////////
+//
+// Graphical primitives
+//
+
+// S COORDINATES
+static void r2d_s_fill_point(T_R2D_GC_PTR gc,INT16 x,INT16 y,INT16 pen_size)
+{
+    INT16 ul_x,ul_y,br_x,br_y,bs,as;
+//    UINT32 f,b;
+
+    bs=(pen_size>>1);
+    as=(pen_size>>1) + (pen_size&1);
+
+	// Squares are not symetrically placed
+	// since pen_size is not always odd
+    // Mirroring must take it into account so that the
+	// asymetry is also mirrored
+#ifdef R2D_MIRRORED_X
+    ul_x=x-bs;
+	br_x=x+as-1;
+#else
+	ul_x=x-bs+1;
+	br_x=x+as;
+    
+#endif
+
+#ifdef R2D_MIRRORED_Y
+    ul_y=y-bs;
+    br_y=y+as-1;
+#else
+    ul_y=y-bs+1;
+    br_y=y+as;
+#endif
+
+    //printf("%d %d %d %d\n",ul_x,ul_y,br_x,br_y);
+
+	
+    r2d_s_fill_rectangle(gc,ul_x-1,ul_y-1,br_x+1,br_y+1,FALSE);
+
+}
+
+// Draw point in gc at position (x,y)
+// S COORDINATES
+static void r2d_s_draw_point(T_R2D_GC_PTR gc,INT16 x,INT16 y,INT32 l)
+{
+  UINT32 pixel_value;
+  INT16 pen_size;
+  BOOLEAN isLCD_format,can_draw;
+  T_R2D_GC* pgc;
+
+  can_draw=FALSE;
+
+  if (r2d_dash_enabled(gc))
+  {
+	  // Q16, 0x0008000 is 8 pixels
+	  if (!(l&0x00080000))
+		  can_draw=TRUE;
+  }
+  else
+	  can_draw=TRUE;
+  
+  if (can_draw==TRUE)
+  {
+  pgc=(T_R2D_GC*)gc;
+
+  isLCD_format=((T_R2D_FRAMEBUFFER*)(((T_R2D_GC*)gc)->p_frame_buffer))->kind!=R2D_FULL_KIND;
+
+
+  pen_size=r2d_get_pen_size(gc);
+
+  
+  if (r2d_point_in_shape(((T_R2D_GC*)gc)->p_s_clipping_shape,x,y))
+  {
+
+	 
+    if (pen_size>1)
+	   r2d_s_fill_point(gc,x,y,pen_size);
+     
+       
+    else
+    {
+       // Dithering is relative to global coordinate system
+      
+
+   // Color is recomputed only when dithering is on
+   #if (R2D_DITHERING == R2D_ON)
+       if (isLCD_format)
+           pixel_value=r2d_get_dithering_matrix_entry(((T_R2D_GC*)gc)->p_foreground_dithered_cache,x,y);
+       else
+           pixel_value=((T_R2D_GC*)gc)->foreground_pixel_value;
+   #else
+       pixel_value=((T_R2D_GC*)gc)->foreground_pixel_value;
+   #endif
+
+       // Low level pixel drawing routine with CONVERTED color value
+	   
+       IND_r2d_write_lcd_pixel(gc,x,y,pixel_value);
+
+    }
+  }
+  
+  r2d_check_and_send_event(gc);
+  }
+}
+
+// Draw point in gc at position (x,y)
+// H COORDINATES
+void r2d_draw_point(T_R2D_GC_PTR gc,INT16 x,INT16 y)
+{
+	if (r2d_point_in_shape(((T_R2D_GC*)gc)->p_clipping_shape,x,y))
+	{
+		r2d_local_to_global(gc,&x,&y);
+	
+#ifdef R2D_MIRRORED_X
+	  x=r2d_get_width(gc)-1-x;
+#endif
+
+#ifdef R2D_MIRRORED_Y
+	  y=r2d_get_height(gc)-1-y;
+#endif
+
+   
+	
+	
+	if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->kind==0)
+      r2d_update_region((T_R2D_GC*)gc,x,y,x,y);
+#if (R2D_REFRESH == R2D_VERTICAL)
+	r2d_s_draw_point(gc,x,y,0);
+#else
+	r2d_s_draw_point(gc,y,x,0);
+#endif
+
+	}
+}
+
+void           r2d_erase_point(T_R2D_GC_PTR gc,INT16 x,INT16 y)
+{
+
+   T_R2D_ARGB_COLOR col;
+
+   
+if (r2d_point_in_shape(((T_R2D_GC*)gc)->p_clipping_shape,x,y))
+{
+	r2d_local_to_global(gc,&x,&y);
+	col=r2d_get_foreground_color(gc);
+   r2d_set_foreground_color(gc,r2d_get_background_color(gc));
+	
+#ifdef R2D_MIRRORED_X
+	  x=r2d_get_width(gc)-1-x;
+#endif
+
+#ifdef R2D_MIRRORED_Y
+	  y=r2d_get_height(gc)-1-y;
+
+#endif
+
+   
+   if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->kind==0)
+      r2d_update_region((T_R2D_GC*)gc,x,y,x,y);
+   
+	#if (R2D_REFRESH == R2D_VERTICAL)
+	r2d_s_draw_point(gc,x,y,0);
+#else
+	r2d_s_draw_point(gc,y,x,0);
+#endif
+
+   r2d_set_foreground_color(gc,col);
+}
+   
+
+}
+
+// Move pen to point (x,y)
+// H_COORDINATES
+void           r2d_moveto(T_R2D_GC_PTR gc,INT16 x,INT16 y)
+{
+   ((T_R2D_GC*)gc)->pen_x=x;
+   ((T_R2D_GC*)gc)->pen_y=y;
+}
+
+
+static BOOLEAN r2d_clipt(INT32 denom,INT32 num,INT32 *tE,INT32 *tL)
+{
+    INT32 t;
+    
+    //printf("clipt %08X %08X\n",denom,num);
+    if (denom<0)
+    {
+       t=r2d_fixed_division(num,denom);
+       //printf("denom<0 ; t=%08X\n",t);
+       if (t>*tL)
+         return FALSE;
+       else if (t > *tE)
+         *tE=t;
+    }
+    else if (denom>0)
+    {
+       t=r2d_fixed_division(num,denom);
+       //printf("denom>0 ; t=%08X\n",t);
+       if (t<*tE)
+         return FALSE;
+       else if (t < *tL)
+          *tL=t;
+    } else if (num<0)
+      return FALSE;
+
+     //printf("t=%08X tE=%08X tL=%08X\n",t,*tE,*tL);
+     return TRUE;
+}
+
+static BOOLEAN r2d_clip_line(T_R2D_SHAPE *self,INT32 *x0,INT32 *y0,INT32 *x1, INT32 *y1)
+{
+	T_R2D_RECTANGLE *r;
+    BOOLEAN visible=FALSE;
+    INT32 dx=*x1-*x0;
+    INT32 dy=*y1-*y0;
+
+	r=(T_R2D_RECTANGLE*)self;
+  
+    // Can only clip against reactangular shapes
+    if (self->p_r2d_class==&r2d_class_rectangle)
+    {
+      
+      if ((dx==0) && (dy==0) && r2d_point_in_shape(self,(*x0>>16) & 0x0FFFF,(*y0>>16) & 0x0FFFF))
+        visible=TRUE;
+      else
+      {
+         INT32 tE=0;
+         INT32 tL=(1<<16);
+
+
+         if (r2d_clipt(dx,((r2d_get_xmax(self)-1)<<16)-*x0,&tE,&tL)==TRUE)
+           if (r2d_clipt(-dx,*x0-(r2d_get_xmin(self)<<16),&tE,&tL)==TRUE)
+             if (r2d_clipt(-dy,*y0-(r2d_get_ymin(self)<<16),&tE,&tL)==TRUE)
+               if (r2d_clipt(dy,((r2d_get_ymax(self)-1)<<16) -*y0,&tE,&tL)==TRUE)
+               {
+                   visible=TRUE;
+
+                   //printf("tE=%08X tL=%08X\n",tE,tL);
+                   // 1 with fixed point representation
+                   if (tL < (1<<16))
+                   {
+                       // dx and dy are integer
+                       // tL and tE should be lower than 1
+                       // So intermediate values are shifted so that
+                       // the product can be done on 32 bits instead of 64 bits
+                       *x1=*x0 + (tL*(dx>>16)) ;
+                       *y1=*y0 + (tL*(dy>>16)) ;
+                       //printf("x1=%08X y1=%08X\n",*x1,*y1);
+                   }
+                   if (tE > 0)
+                   {
+                       *x0 += (tE*(dx>>16)) ;
+                       *y0 += (tE*(dy>>16)) ;
+                       //printf("x0=%08X y0=%08X\n",*x0,*y0);
+
+                   }
+               }
+      }
+    }
+    return(visible);
+}
+
+// 0.5 is rounded down because in line drawing the bottom
+// point is chosen when the line is passing through the middle
+// point
+#define R2D_Q16INT_VALUE(x) ((x+(0x7FFF)) & 0xFFFF0000)
+
+#define R2D_FLOOR_VALUE(x) (x & 0xFFFF0000)
+
+// Positive times signed value
+#define R2D_MULT_PN(x,y) ((((x >> 16) & 0x0FFFF)*y) + (((x & 0x0FFFF) * y) >> 16))
+
+// draw line from pen position to point (x,y)
+void           r2d_lineto(T_R2D_GC_PTR gc,INT16 x,INT16 y)
+{
+    // Clipping
+    // Now drawing with low level function
+
+    UINT32 pixel_value;
+    INT32 P0X,P0Y,P1X,P1Y,c,cy,cx;
+    INT32 current_x,current_y,tmp,tmpx,tmpy;
+    INT32 dx,dy,dn,dnp,dnn;
+    INT16 xincrement,yincrement,reverse;
+    INT32 nx,ny;
+    INT16 pen_size;
+	INT32 l;
+    BOOLEAN isLCD_format;
+	l=0;
+
+	
+
+    isLCD_format=((T_R2D_FRAMEBUFFER*)(((T_R2D_GC*)gc)->p_frame_buffer))->kind!=R2D_FULL_KIND;
+
+    pixel_value=((T_R2D_GC*)gc)->foreground_pixel_value;
+    pen_size=r2d_get_pen_size(gc);
+
+    P0X=((T_R2D_GC*)gc)->pen_x << 16;
+    P0Y=((T_R2D_GC*)gc)->pen_y << 16;
+    P1X=x << 16;
+    P1Y=y << 16;
+    //printf("\n\n(%d,%d) ; (%d,%d)\n",P0X>>16,P0Y>>16,P1X>>16,P1Y>>16);
+    
+// As far as possible, most datas are computed before clipping to get same accuracy
+// Update pen position
+    ((T_R2D_GC*)gc)->pen_x=x;
+    ((T_R2D_GC*)gc)->pen_y=y;
+
+if (r2d_clip_line((((T_R2D_GC*)gc)->p_clipping_shape),&P0X,&P0Y,&P1X,&P1Y))
+{
+
+// Local to global
+
+	P0X-=((T_R2D_GC*)gc)->org_x << 16;
+    P0Y-=((T_R2D_GC*)gc)->org_y << 16;
+    P1X-=((T_R2D_GC*)gc)->org_x << 16;
+    P1Y-=((T_R2D_GC*)gc)->org_y << 16;
+
+	 
+#ifdef R2D_MIRRORED_X
+  P0X=((r2d_get_width(gc)-1)<<16)-P0X;
+  P1X=((r2d_get_width(gc)-1)<<16)-P1X;
+#endif
+
+#ifdef R2D_MIRRORED_Y
+  P0Y=((r2d_get_height(gc)-1)<<16)-P0Y;
+  P1Y=((r2d_get_height(gc)-1)<<16)-P1Y;
+#endif
+
+    
+    /*printf("%d %d %d %d\n",r2d_get_xmin(((T_R2D_GC*)gc)->p_clipping_shape),
+    r2d_get_xmax(((T_R2D_GC*)gc)->p_clipping_shape),
+    r2d_get_ymin(((T_R2D_GC*)gc)->p_clipping_shape),
+    r2d_get_ymax(((T_R2D_GC*)gc)->p_clipping_shape)
+   );*/
+
+    //r2d_clip_line(&(((T_R2D_GC*)gc)->clipping_rectangle),&P0X,&P0Y,&P1X,&P1Y);
+    
+        //printf("(%d,%d) ; (%d,%d)\n",P0X>>16,P0Y>>16,P1X>>16,P1Y>>16);
+        //printf("(%08X,%08X) ; (%08X,%08X)\n------------------\n",P0X,P0Y,P1X,P1Y);
+      
+      //printf("(%d,%d) ; (%d,%d)\n",P0X>>16,P0Y>>16,P1X>>16,P1Y>>16);
+
+if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->kind==0)
+      r2d_update_region((T_R2D_GC*)gc,P0X>>16,P0Y>>16,P1X>>16,P1Y>>16);
+
+
+#if (R2D_REFRESH==R2D_VERTICAL)
+    current_x=P0Y;
+    current_y=P0X;
+    nx=P1Y; ny=P1X;
+
+    
+#else
+    current_x=P0X;
+    current_y=P0Y;
+    nx=P1X; ny=P1Y;
+#endif
+
+    dx=nx-current_x;
+    dy=ny-current_y;
+    
+    yincrement=1;
+
+    if (dx>=0)
+    {
+       xincrement=1;
+    }
+    else
+    {
+       xincrement=-1;
+       dx=-dx;
+    }
+
+    if (dy>=0)
+    {
+       yincrement=1;
+    }
+    else
+    {
+       yincrement=-1;
+       dy=-dy;
+    }
+
+    if (dy<=dx)
+      reverse=0;
+    else
+    {
+      reverse=1;
+      tmp=dy;
+      dy=dx;
+      dx=tmp;
+      tmp=current_y;
+      current_y=current_x;
+      current_x=tmp;
+      tmp=ny;
+      ny=nx;
+      nx=tmp;
+      tmp=yincrement;
+      yincrement=xincrement;
+      xincrement=tmp;
+    }
+
+    reverse=1-reverse;
+
+
+    //printf("dx,dy,current_x,current_y,nx,ny\n%08X %08X %08X %08X %08X %08X\n",dx,dy,current_x,current_y,nx,ny);
+
+
+    // Remember that when one is speaking about a move of 1 horizontally
+    // one is talking about internal coordinates used by the algorithm
+    // In the window it may be vertical or horizontal depending on the
+    // refresh mode and on the line slope (<1 or >1)
+
+
+    dn=dy - (dx >> 1)  ; // d0 = a-b/2 
+    dnp=dy-dx ;
+    dnn=dy;
+    
+    // We draw the point which is the closest to the line so we use INT
+    // instead of FLOOR or CEIL
+    // But, then, the drawing algorithm assumes that the first drawn point is
+    // the lowest one
+
+    // If F(P) = 0 is a parametric description of the line
+    // The, when one is moving from the point of the line on the frontier
+    // of the clipping rectangle to the closest
+    // point with x,y integer coordinate then one needs to compute dx*cy and dy*cx
+    // since it is the variation of F along that path
+    tmpx=(INT32)R2D_Q16INT_VALUE(current_x);
+    tmpy=(INT32)R2D_Q16INT_VALUE(current_y);
+    cx = current_x - tmpx;
+    cy = current_y - tmpy;
+
+
+    c= -R2D_MULT_PN(dx,cy) + R2D_MULT_PN(dy,cx);
+    
+
+    //printf("cx=%08X, cy=%08X, c=%08X, dx=%08X, dy=%08X \n",cx,cy,c,dx,dy);
+
+    // Then, one can add c to the variation due to a move of 1 horizontally
+    // an a move of 1/2 vertically.
+    // One finally get the value of the decision variable at the midpoint
+    dn+=c;
+
+    //printf("%08X %08X %08X %08X %08X\n",current_x,current_y,nx,ny);
+    current_x=((INT32)R2D_Q16INT_VALUE(current_x)) >> 16;
+    current_y=((INT32)R2D_Q16INT_VALUE(current_y)) >> 16;
+    nx=((INT32)R2D_Q16INT_VALUE(nx)) >> 16;
+    ny=((INT32)R2D_Q16INT_VALUE(ny)) >> 16;
+
+    
+    //printf("dx=%08X dy=%08X dn=%08X dnp=%08X dnn=%08X\n",dx,dy,dn,dnp,dnn);
+    
+    if (reverse)
+    {
+        r2d_s_draw_point(gc,current_y,current_x,l);
+        
+    }
+    else
+    {
+        r2d_s_draw_point(gc,current_x,current_y,l);
+    }
+    
+    
+
+    while(current_x != nx)
+    {
+
+      if (dn>0)
+      {
+        dn += dnp; 
+        current_y+=yincrement;
+		l=l+SQRT2;
+      }
+      else if (dn<0)
+      {
+        dn += dnn;
+		l=l+0x10000;
+      } 
+      else
+      {
+          if (xincrement<0)
+          {
+             dn += dnp; 
+             current_y+=yincrement;
+			 l=l+SQRT2;
+          }
+          else
+          {
+             dn += dnn;
+			 l=l+0x10000;
+          }
+      }
+
+      current_x+=xincrement;
+      //printf("    -->%08X %d %d\n",dn,current_x,current_y);
+      // Color is recomputed only when dithering is on
+
+
+      if (reverse)
+      {
+
+          r2d_s_draw_point(gc,current_y,current_x,l);
+      }
+      else
+      {
+          r2d_s_draw_point(gc,current_x,current_y,l);
+      }
+   }
+   } // END CLIPPING
+
+   //printf("END LINE\n");
+   
+  
+   r2d_check_and_send_event(gc);
+}
+
+// Used for filling
+// Do clipping and coordinate transforms then call
+// low levl line filling algorithm
+// No mirroring check since it is a low level routine
+// S COORDINATES
+static void r2d_s_fill_line(T_R2D_GC_PTR gc,INT16 x,INT16 y,INT16 nb,BOOLEAN background)
+{
+   T_R2D_SHAPE_PTR clip;
+   T_R2D_RECTANGLE *r;
+//   INT16 tmp;
+
+   clip=((T_R2D_GC*)gc)->p_s_clipping_shape;
+   r=(T_R2D_RECTANGLE*)clip;
+
+
+   if ((x>=r2d_get_xmin(clip)) && ( x<r2d_get_xmax(clip)))
+   {
+      if (y<r2d_get_ymin(clip))
+      {
+       nb-=r2d_get_ymin(clip)-y;
+       y=r2d_get_ymin(clip);
+      }
+
+      if (y+nb-1>=r2d_get_ymax(clip))
+        nb=r2d_get_ymax(clip)-y;
+
+      if (nb>0)
+
+        IND_r2d_write_lcd_line(gc,x,y,nb,background); 
+   }
+
+
+}
+
+// Used for filling
+// Do clipping and coordinate transforms then call
+// low levl line filling algorithm
+// No mirroring check since it is a low level routine
+// S COORDINATES
+static void r2d_s_arc_fill_line(T_R2D_GC_PTR gc,INT16 x,INT16 y,
+								INT16 org_x,INT16 org_y,INT16 nb,T_R2D_ARC_REGION *rgn,BOOLEAN background)
+{
+   T_R2D_SHAPE_PTR clip;
+   T_R2D_RECTANGLE *r;
+//   INT16 tmp;
+
+   clip=((T_R2D_GC*)gc)->p_s_clipping_shape;
+   r=(T_R2D_RECTANGLE*)clip;
+
+
+   if ((x>=r2d_get_xmin(clip)) && ( x<r2d_get_xmax(clip)))
+   {
+      if (y<r2d_get_ymin(clip))
+      {
+       nb-=r2d_get_ymin(clip)-y;
+       y=r2d_get_ymin(clip);
+      }
+
+      if (y+nb-1>=r2d_get_ymax(clip))
+        nb=r2d_get_ymax(clip)-y;
+
+      if (nb>0)
+
+        r2d_arc_write_lcd_line(gc,x,y,org_x,org_y,nb,rgn,background); 
+   }
+
+
+}
+
+// S coordinates
+static void           r2d_s_fill_rectangle(T_R2D_GC_PTR gc,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y,BOOLEAN background)
+{
+
+    INT16 i;
+	// The ifdef is not passing the coordinates to S frame.
+	// It just changing the way the square is scanned
+	// to improve behavior relatively to the scanning mode.
+       for(i=ul_x+1;i<=br_x-1;i++)
+       {
+           r2d_s_fill_line(gc,i,ul_y+1,(br_y-ul_y)+1-2,background);
+       }
+     
+}
+
+// Background allows to chose the color for filling
+// If TRUE, background color is used
+// (required for pen_size different from 1)
+// L coordinates
+static void           r2d_df_rectangle(T_R2D_GC_PTR gc,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y,BOOLEAN background)
+{
+
+    INT16 i;
+	// The ifdef is not passing the coordinates to S frame.
+	// It just changing the way the square is scanned
+	// to improve behavior relatively to the scanning mode.
+     #if (R2D_REFRESH==R2D_VERTICAL)
+       for(i=ul_x+1;i<=br_x-1;i++)
+       {
+           r2d_s_fill_line(gc,i,ul_y+1,(br_y-ul_y)+1-2,background);
+       }
+     #else
+       for(i=ul_y+1;i<=br_y-1;i++)
+       {
+           r2d_s_fill_line(gc,i,ul_x+1,(br_x-ul_x)+1-2,background);
+       }
+     #endif
+}
+
+void           r2d_draw_rectangle(T_R2D_GC_PTR gc,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y)
+{
+
+	 
+     r2d_moveto(gc,ul_x,ul_y);
+     r2d_lineto(gc,br_x,ul_y);
+
+	 
+     r2d_moveto(gc,br_x,ul_y+1);
+     r2d_lineto(gc,br_x,br_y);
+
+	 
+     r2d_moveto(gc,br_x-1,br_y);
+     r2d_lineto(gc,ul_x,br_y);
+
+	 
+	 r2d_moveto(gc,ul_x,ul_y+1);
+     r2d_lineto(gc,ul_x,br_y-1);
+	 
+}
+
+void           r2d_fill_rectangle(T_R2D_GC_PTR gc,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y)
+{
+#ifdef R2D_MIRRORED_X
+	INT16 tmp;
+#endif
+	r2d_local_to_global(gc,&ul_x,&ul_y);
+	r2d_local_to_global(gc,&br_x,&br_y);
+
+	
+#ifdef R2D_MIRRORED_X
+	   ul_x = r2d_get_width(gc)-1-ul_x;
+	   br_x = r2d_get_width(gc)-1-br_x;
+       R2D_SWAP(ul_x,br_x);
+#endif
+
+#ifdef R2D_MIRRORED_Y
+	   ul_y = r2d_get_height(gc)-1-ul_y;
+	   br_y = r2d_get_height(gc)-1-br_y;	   
+	   R2D_SWAP(ul_y,br_y);
+#endif
+       r2d_df_rectangle(gc,ul_x,ul_y,br_x,br_y,TRUE);
+	   if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->kind==0)
+          r2d_update_region((T_R2D_GC*)gc,ul_x,ul_y,br_x,br_y);
+
+   r2d_check_and_send_event(gc);
+}
+
+
+
+void           r2d_df_circle(T_R2D_GC_PTR gc,
+                 INT16 x,INT16 y,INT16 r,BOOLEAN filling)
+{
+  INT32 current_x,current_y,my,mx;
+  INT32 dn,dnp,dnn,org_x,org_y,limit,rs,tmp,D_yx;
+  INT32 l;
+  INT32 p; // perimeter divided by 4
+
+  l=0;
+
+  p=(PI*r)>>1;
+  
+
+  #if (R2D_REFRESH==R2D_VERTICAL)
+    current_x=y;
+    current_y=x+r;
+    org_x=y; 
+    org_y=x;
+	
+  #else
+    current_x=x;
+    current_y=y+r;
+    org_x=x;
+    org_y=y;
+  #endif
+
+  D_yx=org_y-org_x;
+
+  dnp=((-r) << 1) + 5;
+  dnn=3;
+  dn=-r;
+  rs=r*r;
+  limit=-r;
+
+ 
+
+
+
+  tmp=1;
+  while(limit<=0)
+  {
+
+     // Some points are their own image with symetries
+     // and yet they must not be drawn twice for drawing
+     // modes other than copy hence the tests
+
+     my=(org_y << 1) - current_y;
+     mx=(org_x << 1) - current_x;
+
+     // PY or PX is reflection relatively to horizontal line or vertical line
+     // PY = vertical mirroring where bottom becomes top
+
+     // D = diagonal mirroring
+
+     // AB means apply transform A then B
+
+     // I is identity
+
+     // Everything is done relatively to what is displayed on the LCD
+     // so my may be PX according to the refresh mode
+
+
+     if (!filling)
+     {
+        if (mx != current_x)
+        {
+          r2d_s_draw_point(gc,current_y,current_x,l); // I : 3
+          r2d_s_draw_point(gc,mx+D_yx,current_y-D_yx,l); // D PX : 5
+          r2d_s_draw_point(gc,my,mx,l); // PX.PY : 7
+          r2d_s_draw_point(gc,current_x+D_yx,my-D_yx,l); // D PY : 1
+        }
+
+        if (current_x != current_y)
+        {
+          r2d_s_draw_point(gc,current_x+D_yx,current_y-D_yx,l+p); // D : 4
+          r2d_s_draw_point(gc,current_y,mx,l+p); // PY : 2
+        }
+
+        if ((mx != my) || (current_x != current_y))
+          r2d_s_draw_point(gc,my,current_x,l+p); // PX : 6
+
+
+        if (mx != my)
+          r2d_s_draw_point(gc,mx+D_yx,my-D_yx,l+p); // D PX PY : 8
+
+     }
+     else
+     {
+         //printf("%d\n",current_x);
+         if (tmp)
+         {
+           if (current_x-mx!=0)
+           {
+             r2d_s_fill_line(gc,my,mx+1,(current_x-mx)+1-2,TRUE);
+             r2d_s_fill_line(gc,current_y,mx+1,(current_x-mx)+1-2,TRUE);
+           }
+         }
+         if (mx != my) 
+         {
+           r2d_s_fill_line(gc,mx+D_yx,my+1-D_yx,(current_y-my)+1-2,TRUE);
+           if (current_x != mx);
+            r2d_s_fill_line(gc,current_x+D_yx,my+1-D_yx,(current_y-my)+1-2,TRUE);
+         }
+     }
+
+
+     if (dn>0) // M outside circle, point below must be drawn
+     {
+        dn+=dnp;
+        dnp+=4;
+        
+        current_y--;
+		l+=SQRT2;
+        limit+=2;
+        tmp=1;
+     }
+     else
+     {
+        dn+=dnn;
+        dnp+=2;
+        limit+=1;
+        tmp=0;
+		l+=0x10000;
+     }
+     
+     current_x++;
+     dnn+=2;
+
+     
+     
+     
+  }
+  //printf("%08X\n",limit);
+}
+
+void           r2d_draw_circle(T_R2D_GC_PTR gc,
+                 INT16 x,INT16 y,INT16 r)
+{
+
+	r2d_local_to_global(gc,&x,&y);
+  
+
+#ifdef R2D_MIRRORED_X
+	   x = r2d_get_width(gc)-1-x;
+#endif
+
+#ifdef R2D_MIRRORED_Y
+	   y = r2d_get_height(gc)-1-y;
+#endif
+
+
+   r2d_df_circle(gc,x,y,r,FALSE);
+   if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->kind==0)
+          r2d_update_region((T_R2D_GC*)gc,x-r,y-r,x+r,y+r);
+   r2d_check_and_send_event(gc);
+}
+
+void           r2d_fill_circle(T_R2D_GC_PTR gc,
+                 INT16 x,INT16 y,INT16 r)
+{
+	r2d_local_to_global(gc,&x,&y);
+
+#ifdef R2D_MIRRORED_X
+	    x = r2d_get_width(gc)-1-x;
+#endif
+
+#ifdef R2D_MIRRORED_Y
+	   y = r2d_get_height(gc)-1-y;
+#endif
+
+   r2d_df_circle(gc,x,y,r,TRUE);
+   if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->kind==0)
+          r2d_update_region((T_R2D_GC*)gc,x-r,y-r,x+r,y+r);
+   r2d_check_and_send_event(gc);
+}
+
+
+
+// h and v are used to draw roundrectangle (each piece of the ellipse must be
+// separated from other parts)
+void           r2d_df_ellipse(T_R2D_GC_PTR gc,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y, INT16 h,INT16 v,BOOLEAN filling)
+{
+  INT32 current_x,current_y,limit,newx,newy,org_x,org_y;
+  INT32 dn,dnp,dnn,as,bs,tmp,a,b,mx,my,lastx,lasty,round;
+  INT32 l;
+  l=0;
+
+  if ((h!=0) || (v!=0))
+   round=1;
+  else
+   round=0;
+
+
+  // Width of rectangle is assumed to be even
+  // and it is forced to even dimension
+
+
+  #if (R2D_REFRESH==R2D_VERTICAL)
+    org_x=(ul_y + br_y)>>1;
+    org_y=(br_x + ul_x)>>1; 
+    a=(br_y-ul_y)>>1;
+    as=R2D_SQUARE_P(a);
+    b=(br_x-ul_x)>>1;
+    bs=R2D_SQUARE_P(b);
+
+    current_x=org_x ;
+    current_y=org_y+b; 
+    newx=org_y;
+    newy=org_x+a;
+    
+  #else
+    org_x=(ul_x + br_x)>>1; 
+    org_y=(br_y + ul_y)>>1;
+
+    b=(br_y-ul_y)>>1;
+    bs=R2D_SQUARE_P(b);
+    a=(br_x-ul_x)>>1;
+    as=R2D_SQUARE_P(a);
+
+    current_x=org_x;
+    current_y=org_y+b; 
+    newx=org_y;
+    newy=org_x+a;
+	R2D_SWAP(h,v);
+    
+  #endif
+
+  // Now, shift by 2 to have the accuracy required by division by four
+
+
+  //printf("current_x=%d, current_y=%d,as=%08X bs=%08X\n",current_x,current_y,as,bs);
+
+
+  // New shift by 2 because one needs to compute quarter of values
+  bs<<=2; as <<=2;b<<=2;a<<=2;
+
+  limit = - ((as * b) >> 2); // -a^2 * b in Q2
+
+  tmp=-((as * b) >> 2);
+  dnp=(bs << 1) + bs + (tmp<<1) + (as << 1);
+  dnn=(bs << 1) + bs;
+  dn=bs + tmp + (as >>2);
+  
+  
+  
+  //printf("dn=%f dnp=%f dnn=%f\n",dn / 8.0,dnp / 8.0,dnn / 8.0);
+
+/*#if (R2D_REFRESH==R2D_VERTICAL)
+  r2d_draw_point(gc,current_y,current_x);
+#else
+  r2d_draw_point(gc,current_x,current_y);
+#endif*/
+  
+  //printf("as=%08X bs=%08X\n",as,bs);
+
+  tmp=1;
+  while(limit<=0)
+  {
+     my=(org_y << 1) - current_y;
+     mx=(org_x << 1) - current_x;
+
+     lastx=current_y;
+     lasty=current_x;
+
+     if (!filling)
+     {
+       r2d_s_draw_point(gc,(current_y)+h,(current_x)+v,l); // I : 1
+       r2d_s_draw_point(gc,my,mx,l); // PX PY : 2
+
+       if ((mx != current_x) || (round))
+       {
+          r2d_s_draw_point(gc,(current_y)+h,mx,l); // PY : 3
+          r2d_s_draw_point(gc,my,current_x+v,l); // PX : 4
+       }
+     }
+     else
+     {
+         if (tmp)
+         {
+
+           // Used to remove point on horizontal extremity
+           if (current_x+v-mx !=0)
+           {
+             r2d_s_fill_line(gc,my,mx+1,(current_x+v-mx)+1-2,TRUE);
+             r2d_s_fill_line(gc,current_y+h,mx+1,(current_x+v-mx)+1-2,TRUE);
+           }
+         }  
+     }
+
+     if (dn>=0) // M outside circle, point below must be drawn
+     {
+        dn+=dnp;
+        dnp+=((bs + as) << 1);
+        current_y--; 
+        limit+=(as + bs);
+        tmp=1;
+		l+=SQRT2;
+     }
+     else
+
+     {
+        dn+=dnn;
+        dnp+=(bs << 1);
+        limit+=bs;
+        tmp=0;
+		l+=0x10000;
+        
+     }
+     current_x++; 
+     dnn+=(bs << 1);
+
+     
+     //printf("limit=%f, dn=%f,dnn=%f,dnp=%f, pos= %d %d\n",limit / 4.0,dn / 4.0,dnn / 4.0,dnp /
+//4.0,current_y,current_x);
+     
+     
+  }
+  
+  current_x=newx;
+  current_y=newy;
+  R2D_SWAP(as,bs);
+  R2D_SWAP(a,b);
+  
+  limit = - ((as * b) >> 2);
+
+  tmp=-((as * b) >> 2);
+  dnp=(bs << 1) + bs + (tmp<<1) + (as << 1);
+  dnn=(bs << 1) + bs;
+  dn=bs + tmp + (as >>2);
+
+  //printf("newx=%d, newy=%d,a=%d,b=%d\n",newx,newy,a,b);
+  //printf("dn=%f dnp=%f dnn=%f\n",dn / 4.0,dnp / 4.0,dnn / 4.0);
+
+  tmp=1;
+
+ /* #if (R2D_REFRESH==R2D_VERTICAL)
+    r2d_draw_point(gc,current_x,current_y);
+  #else
+    r2d_draw_point(gc,current_y,current_x);
+  #endif*/
+
+  while(limit<0)
+  {
+     
+     my=(org_x << 1) - current_y;
+     mx=(org_y << 1) - current_x;
+
+     if (!filling)
+     {
+        if ((lastx!=current_x) || (lasty!=current_y))
+        {
+
+          r2d_s_draw_point(gc,(current_x)+h,(current_y)+v,l); // D : 5
+
+
+          r2d_s_draw_point(gc,mx,my,l); // D PX PY : 6
+
+          if ((mx != (current_x+h)) || (round))
+          {
+            r2d_s_draw_point(gc,(current_x)+h,my,l); // D PY : 7
+            r2d_s_draw_point(gc,mx,(current_y)+v,l); // D PX : 8
+          }
+
+        }
+     }
+     else
+     {
+        if (current_x != lastx)
+        {
+           r2d_s_fill_line(gc,mx,my+1,(current_y+v-my)+1-2,TRUE);
+           if ((current_x+h) !=mx)
+             r2d_s_fill_line(gc,current_x+h,my+1,(current_y+v-my)+1-2,TRUE);
+        }
+     }
+
+     if (dn>0) // M outside circle, point below must be drawn
+     {
+        dn+=dnp;
+        dnp+=((bs + as) << 1);
+        current_y--; // Q1
+        limit+=(as + bs);
+        tmp=1;
+		l+=SQRT2;
+     }
+     else
+     {
+        dn+=dnn;
+        dnp+=(bs << 1);
+        limit+=bs;
+        tmp=0;
+		l+=0x10000;
+        
+     }
+     current_x++; // Q1
+     dnn+=(bs << 1);
+
+     
+     
+     
+     //printf("limit=%f, dn=%f,dnn=%f,dnp=%f, pos= %d %d\n",limit / 4.0,dn / 4.0,dnn / 4.0,dnp / 4.0,current_y,current_x);
+     
+     
+  }
+}
+
+void           r2d_draw_ellipse(T_R2D_GC_PTR gc,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y)
+{
+#ifdef R2D_MIRRORED_X
+	INT16 tmp;
+#endif
+	r2d_local_to_global(gc,&ul_x,&ul_y);
+	r2d_local_to_global(gc,&br_x,&br_y);
+	
+
+#ifdef R2D_MIRRORED_X
+	   ul_x = r2d_get_width(gc)-1-ul_x;
+	   br_x = r2d_get_width(gc)-1-br_x;
+	   R2D_SWAP(ul_x,br_x);
+#endif
+
+#ifdef R2D_MIRRORED_Y
+	   ul_y = r2d_get_height(gc)-1-ul_y;
+	   br_y = r2d_get_height(gc)-1-br_y;
+	   R2D_SWAP(ul_y,br_y);
+#endif
+
+   r2d_df_ellipse(gc,ul_x,ul_y,br_x,br_y,0,0,FALSE);
+   if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->kind==0)
+          r2d_update_region((T_R2D_GC*)gc,ul_x,ul_y,br_x,br_y);
+
+   r2d_check_and_send_event(gc);
+}
+
+void           r2d_fill_ellipse(T_R2D_GC_PTR gc,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y)
+{
+#ifdef R2D_MIRRORED_X
+	INT16 tmp;
+#endif
+	r2d_local_to_global(gc,&ul_x,&ul_y);
+	r2d_local_to_global(gc,&br_x,&br_y);
+
+	
+#ifdef R2D_MIRRORED_X
+	   ul_x = r2d_get_width(gc)-1-ul_x;
+	   br_x = r2d_get_width(gc)-1-br_x;
+       R2D_SWAP(ul_x,br_x);
+#endif
+
+#ifdef R2D_MIRRORED_Y
+	   ul_y = r2d_get_height(gc)-1-ul_y;
+	   br_y = r2d_get_height(gc)-1-br_y;
+	   R2D_SWAP(ul_y,br_y);
+#endif
+
+   r2d_df_ellipse(gc,ul_x,ul_y,br_x,br_y,0,0,TRUE);
+   if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->kind==0)
+          r2d_update_region((T_R2D_GC*)gc,ul_x,ul_y,br_x,br_y);
+
+   r2d_check_and_send_event(gc);
+}
+
+void           r2d_draw_round_rectangle(T_R2D_GC_PTR gc,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y,INT16 h,INT16 v)
+{
+#ifdef R2D_MIRRORED_X
+	INT16 tmp;
+#endif
+	r2d_moveto(gc,ul_x+h+1,ul_y);
+  r2d_lineto(gc,br_x-h-1,ul_y);
+
+  r2d_moveto(gc,ul_x+h+1,br_y);
+  r2d_lineto(gc,br_x-h-1,br_y);
+
+  r2d_moveto(gc,ul_x,ul_y+v+1);
+  r2d_lineto(gc,ul_x,br_y-v-1);
+
+  r2d_moveto(gc,br_x,ul_y+v+1);
+  r2d_lineto(gc,br_x,br_y-v-1);
+
+   r2d_local_to_global(gc,&ul_x,&ul_y);
+   r2d_local_to_global(gc,&br_x,&br_y);
+
+	
+#ifdef R2D_MIRRORED_X
+	   ul_x = r2d_get_width(gc)-1-ul_x;
+	   br_x = r2d_get_width(gc)-1-br_x;
+       R2D_SWAP(ul_x,br_x);
+#endif
+
+#ifdef R2D_MIRRORED_Y
+	   ul_y = r2d_get_height(gc)-1-ul_y;
+	   br_y = r2d_get_height(gc)-1-br_y;
+	   R2D_SWAP(ul_y,br_y);  
+#endif
+  r2d_df_ellipse(gc,ul_x,ul_y,ul_x+(h<<1),ul_y+(v<<1),(br_x - ul_x) - (h<<1),(br_y - ul_y) - (v<<1),FALSE);
+if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->kind==0)
+          r2d_update_region((T_R2D_GC*)gc,ul_x,ul_y,br_x,br_y);
+
+  //printf("%d %d\n",ul_x+h+1,br_x-h-1);
+  
+  
+  r2d_check_and_send_event(gc);
+}
+
+void           r2d_fill_round_rectangle(T_R2D_GC_PTR gc,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y,INT16 h,INT16 v)
+{
+#ifdef R2D_MIRRORED_X
+	INT16 tmp;
+#endif
+	r2d_local_to_global(gc,&ul_x,&ul_y);
+   r2d_local_to_global(gc,&br_x,&br_y);
+
+#ifdef R2D_MIRRORED_X
+	   ul_x = r2d_get_width(gc)-1-ul_x;
+	   br_x = r2d_get_width(gc)-1-br_x;
+       R2D_SWAP(ul_x,br_x);
+#endif
+
+
+#ifdef R2D_MIRRORED_Y
+	   ul_y = r2d_get_height(gc)-1-ul_y;
+	   br_y = r2d_get_height(gc)-1-br_y;
+	   R2D_SWAP(ul_y,br_y);
+#endif
+
+  r2d_df_ellipse(gc,ul_x,ul_y,ul_x+(h<<1),ul_y+(v<<1),(br_x - ul_x) - (h<<1),(br_y - ul_y) - (v<<1),TRUE);
+#if (R2D_REFRESH == R2D_VERTICAL)
+  r2d_df_rectangle(gc,ul_x+h,ul_y,br_x-h+1,br_y,TRUE);
+
+#else
+  
+  r2d_df_rectangle(gc,ul_x,ul_y+v,br_x,br_y-v+1,TRUE);
+#endif
+  if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->kind==0)
+          r2d_update_region((T_R2D_GC*)gc,ul_x,ul_y,br_x,br_y);
+  //printf("%d %d\n",ul_x+h+1,br_x-h-1);
+  /*r2d_moveto(gc,ul_x+h+1,ul_y);
+  r2d_lineto(gc,br_x-h-1,ul_y);
+
+  r2d_moveto(gc,ul_x+h+1,br_y);
+  r2d_lineto(gc,br_x-h-1,br_y);
+
+  r2d_moveto(gc,ul_x,ul_y+v+1);
+  r2d_lineto(gc,ul_x,br_y-v-1);
+
+  r2d_moveto(gc,br_x,ul_y+v+1);
+  r2d_lineto(gc,br_x,br_y-v-1);*/
+  r2d_check_and_send_event(gc);
+}
+
+// angle must be between 0 and 359
+static INT32 r2d_cos(INT16 angle)
+{
+	
+
+	if (angle<=90)
+		return(r2d_cos_table[angle]);
+	else if (angle<=180)
+		return(-r2d_cos_table[180-angle]);
+	else if (angle<=270)
+         return(-r2d_cos_table[angle-180]);
+	else
+         return(r2d_cos_table[360-angle]);
+}
+
+// angle must be between 0 and 359
+static INT32 r2d_sin(INT16 angle)
+{
+	angle=angle+90;
+	if (angle>=360)
+		angle=angle-360;
+
+	return(-r2d_cos(angle));
+}
+
+static void r2d_check_angles(INT16 *start,INT16 *stop)
+{
+//   INT16 tmp;
+   
+
+   while(*start<0)
+	   *start+=360;
+   while(*start>=360)
+	   *start-=360;
+
+   while(*stop<0)
+	   *stop+=360;
+   while(*stop>=360)
+	   *stop-=360;
+
+   //if (*start>*stop)
+   //{
+//	   R2D_SWAP((*start),(*stop));
+	  
+   //}
+
+}
+
+
+
+void           r2d_df_arc(T_R2D_GC_PTR gc,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y,T_R2D_ARC_REGION *rgn ,BOOLEAN filling)
+{
+  INT32 current_x,current_y,limit,newx,newy,org_x,org_y;
+  INT32 dn,dnp,dnn,as,bs,tmp,a,b,mx,my,lastx,lasty;
+  INT32 l;
+
+  l=0;
+
+  // Width of rectangle is assumed to be even
+  // and it is forced to even dimension
+
+  if (rgn->is_null)
+	  goto end_arc;
+
+  #if (R2D_REFRESH==R2D_VERTICAL)
+    org_x=(ul_y + br_y)>>1;
+    org_y=(br_x + ul_x)>>1; 
+    a=(br_y-ul_y)>>1;
+    as=R2D_SQUARE_P(a);
+    b=(br_x-ul_x)>>1;
+    bs=R2D_SQUARE_P(b);
+
+    current_x=org_x ;
+    current_y=org_y+b; 
+    newx=org_y;
+    newy=org_x+a;
+
+    
+    
+  #else
+    org_x=(ul_x + br_x)>>1; 
+    org_y=(br_y + ul_y)>>1;
+
+    b=(br_y-ul_y)>>1;
+    bs=R2D_SQUARE_P(b);
+    a=(br_x-ul_x)>>1;
+    as=R2D_SQUARE_P(a);
+
+    current_x=org_x;
+    current_y=org_y+b; 
+    newx=org_y;
+    newy=org_x+a;
+	
+    r2d_arc_region_diagonal_reflect(rgn);
+	r2d_arc_region_x_reflect(rgn);
+	r2d_arc_region_y_reflect(rgn);
+  #endif
+
+	
+	
+	// Orthogonal is (eb,-ea)
+
+	//r2d_s_draw_point(gc,org_y,org_x);
+	//r2d_s_draw_point(gc,((rgn->sa)>>16)+org_y,((rgn->sb)>>16)+org_x);
+	//r2d_s_draw_point(gc,((rgn->sb)>>16)+org_y,((-rgn->sa)>>16)+org_x);
+
+	//r2d_s_draw_point(gc,((rgn->ea)>>16)+org_y,((rgn->eb)>>16)+org_x);
+	
+	//goto end_arc;
+
+  // Now, shift by 2 to have the accuracy required by division by four
+
+
+  //printf("current_x=%d, current_y=%d,as=%08X bs=%08X\n",current_x,current_y,as,bs);
+
+
+  // New shift by 2 because one needs to compute quarter of values
+  bs<<=2; as <<=2;b<<=2;a<<=2;
+
+  limit = - ((as * b) >> 2); // -a^2 * b in Q2
+
+  tmp=-((as * b) >> 2);
+  dnp=(bs << 1) + bs + (tmp<<1) + (as << 1);
+  dnn=(bs << 1) + bs;
+  dn=bs + tmp + (as >>2);
+  
+  
+  
+  //printf("dn=%f dnp=%f dnn=%f\n",dn / 8.0,dnp / 8.0,dnn / 8.0);
+
+/*#if (R2D_REFRESH==R2D_VERTICAL)
+  r2d_draw_point(gc,current_y,current_x);
+#else
+  r2d_draw_point(gc,current_x,current_y);
+#endif*/
+  
+  //printf("as=%08X bs=%08X\n",as,bs);
+
+  tmp=1;
+  while(limit<=0)
+  {
+     my=(org_y << 1) - current_y;
+     mx=(org_x << 1) - current_x;
+
+     lastx=current_y;
+     lasty=current_x;
+
+     if (!filling)
+     {
+	   if (r2d_check_is_in_arc_region(current_y-org_y,current_x-org_x,rgn))
+         r2d_s_draw_point(gc,(current_y),(current_x),l); // I : 1
+
+	   if (r2d_check_is_in_arc_region(my-org_y,mx-org_x,rgn))
+          r2d_s_draw_point(gc,my,mx,l); // PX PY : 2
+
+       if ((mx != current_x) )
+       {
+		  if (r2d_check_is_in_arc_region(current_y-org_y,mx-org_x,rgn))
+             r2d_s_draw_point(gc,current_y,mx,l); // PY : 3
+
+		  if (r2d_check_is_in_arc_region(my-org_y,current_x-org_x,rgn))
+             r2d_s_draw_point(gc,my,current_x,l); // PX : 4
+       }
+     }
+     else
+     {
+         if (tmp)
+         {
+
+           // Used to remove point on horizontal extremity
+           if (current_x-mx !=0)
+           {
+			 r2d_s_arc_fill_line(gc,my,mx+1,org_y,org_x,(current_x-mx)+1-2
+				 ,rgn,TRUE);
+
+             r2d_s_arc_fill_line(gc,current_y,mx+1,org_y,org_x,(current_x-mx)+1-2
+				 ,rgn,TRUE);
+           }
+         }  
+     }
+
+     if (dn>=0) // M outside circle, point below must be drawn
+     {
+        dn+=dnp;
+        dnp+=((bs + as) << 1);
+        current_y--; 
+        limit+=(as + bs);
+        tmp=1;
+		l+=SQRT2;
+     }
+     else
+
+     {
+        dn+=dnn;
+        dnp+=(bs << 1);
+        limit+=bs;
+        tmp=0;
+        l+=0x10000;
+     }
+     current_x++; 
+     dnn+=(bs << 1);
+
+     
+     //printf("limit=%f, dn=%f,dnn=%f,dnp=%f, pos= %d %d\n",limit / 4.0,dn / 4.0,dnn / 4.0,dnp /
+//4.0,current_y,current_x);
+     
+     
+  }
+  
+  current_x=newx;
+  current_y=newy;
+  R2D_SWAP(as,bs);
+  R2D_SWAP(a,b);
+  
+  limit = - ((as * b) >> 2);
+
+  tmp=-((as * b) >> 2);
+  dnp=(bs << 1) + bs + (tmp<<1) + (as << 1);
+  dnn=(bs << 1) + bs;
+  dn=bs + tmp + (as >>2);
+
+  //printf("newx=%d, newy=%d,a=%d,b=%d\n",newx,newy,a,b);
+  //printf("dn=%f dnp=%f dnn=%f\n",dn / 4.0,dnp / 4.0,dnn / 4.0);
+
+  tmp=1;
+
+ /* #if (R2D_REFRESH==R2D_VERTICAL)
+    r2d_draw_point(gc,current_x,current_y);
+  #else
+    r2d_draw_point(gc,current_y,current_x);
+  #endif*/
+
+  while(limit<0)
+  {
+     
+     my=(org_x << 1) - current_y;
+     mx=(org_y << 1) - current_x;
+
+     if (!filling)
+     {
+        if ((lastx!=current_x) || (lasty!=current_y))
+        {
+
+		  if (r2d_check_is_in_arc_region(current_x-org_y,current_y-org_x,rgn))
+             r2d_s_draw_point(gc,(current_x),(current_y),l); // D : 5
+
+
+		  if (r2d_check_is_in_arc_region(mx-org_y,my-org_x,rgn))
+            r2d_s_draw_point(gc,mx,my,l); // D PX PY : 6
+
+          if ((mx != current_x) )
+          {
+			if (r2d_check_is_in_arc_region(current_x-org_y,my-org_x,rgn))
+              r2d_s_draw_point(gc,(current_x),my,l); // D PY : 7
+
+			if (r2d_check_is_in_arc_region(mx-org_y,current_y-org_x,rgn))
+              r2d_s_draw_point(gc,mx,(current_y),l); // D PX : 8
+          }
+
+        }
+     }
+     else
+     {
+        if (current_x != lastx)
+        {
+		   r2d_s_arc_fill_line(gc,mx,my+1,org_y,org_x,(current_y-my)+1-2
+			   ,rgn,TRUE);
+           
+		   if (current_x !=mx)
+             r2d_s_arc_fill_line(gc,current_x,my+1,org_y,org_x,(current_y-my)+1-2
+			 ,rgn,TRUE);
+        }
+     }
+
+     if (dn>0) // M outside circle, point below must be drawn
+     {
+        dn+=dnp;
+        dnp+=((bs + as) << 1);
+        current_y--; // Q1
+        limit+=(as + bs);
+        tmp=1;
+		l+=SQRT2;
+     }
+     else
+     {
+        dn+=dnn;
+        dnp+=(bs << 1);
+        limit+=bs;
+        tmp=0;
+		l+=0x10000;
+        
+     }
+     current_x++; // Q1
+     dnn+=(bs << 1);
+
+     
+     
+     
+     //printf("limit=%f, dn=%f,dnn=%f,dnp=%f, pos= %d %d\n",limit / 4.0,dn / 4.0,dnn / 4.0,dnp / 4.0,current_y,current_x);
+     
+     
+  }
+
+  end_arc:  current_x=0;
+}
+
+void           r2d_draw_arc(T_R2D_GC_PTR gc,INT16 start_angle,INT16 stop_angle,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y)
+{
+#ifdef R2D_MIRRORED_X
+	INT16 tmp;
+#endif
+	BOOLEAN large=FALSE;
+	T_R2D_ARC_REGION rgn;
+	INT32 sa,sb,ea,eb;
+	INT32 a,b;
+
+	a=(br_x-ul_x)>>1;
+	b=(br_y-ul_y)>>1;
+
+	r2d_local_to_global(gc,&ul_x,&ul_y);
+	r2d_local_to_global(gc,&br_x,&br_y);
+
+    rgn.one_sector=0;
+	if (stop_angle-start_angle>180)
+		rgn.one_sector=1;
+
+	if (stop_angle-start_angle==360)
+         rgn.one_sector=2;
+
+	if (start_angle==stop_angle)
+		rgn.is_null=TRUE;
+	else
+		rgn.is_null=FALSE;
+	r2d_check_angles(&start_angle,&stop_angle);
+	
+
+	sa=a * r2d_cos(start_angle); // Q16
+	sb=  -b* r2d_sin(start_angle); // Q16
+	// Orthogonal is (sb,-sa)
+
+	ea= a * r2d_cos(stop_angle); // Q16
+	eb= -b* r2d_sin(stop_angle); // Q16
+	// Orthogonal is (-sb,sa)
+
+	//r2d_moveto(gc,ul_x+a,ul_y+b);
+	//r2d_lineto(gc,ul_x+a+(sa>>16),ul_y+b+(sb>>16));
+
+	// Orthogonal
+	//r2d_moveto(gc,ul_x+a,ul_y+b);
+	//r2d_lineto(gc,ul_x+a+(sb>>16),ul_y+b+((-sa)>>16));
+
+	//r2d_moveto(gc,ul_x+a,ul_y+b);
+	//r2d_lineto(gc,ul_x+a+(ea>>16),ul_y+b+(eb>>16));
+
+	rgn.sa=sa;
+	rgn.sb=sb;
+
+
+	rgn.ea=-ea;
+	rgn.eb=-eb;
+
+	
+	
+
+	
+
+#ifdef R2D_MIRRORED_X
+	   ul_x = r2d_get_width(gc)-1-ul_x;
+	   br_x = r2d_get_width(gc)-1-br_x;
+	   R2D_SWAP(ul_x,br_x);
+
+	   r2d_arc_region_x_reflect(&rgn);
+	   
+#endif
+
+#ifdef R2D_MIRRORED_Y
+	   ul_y = r2d_get_height(gc)-1-ul_y;
+	   br_y = r2d_get_height(gc)-1-br_y;
+	   R2D_SWAP(ul_y,br_y);
+
+	   r2d_arc_region_y_reflect(&rgn);
+#endif
+
+   
+
+   r2d_df_arc(gc,ul_x,ul_y,br_x,br_y,&rgn,FALSE);
+   if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->kind==0)
+          r2d_update_region((T_R2D_GC*)gc,ul_x,ul_y,br_x,br_y);
+
+   r2d_check_and_send_event(gc);
+}
+
+void           r2d_fill_arc(T_R2D_GC_PTR gc, INT16 start_angle,INT16 stop_angle,
+                  INT16 ul_x,INT16 ul_y,INT16 br_x, INT16 br_y)
+{
+#ifdef R2D_MIRRORED_X
+	INT16 tmp;
+#endif
+	BOOLEAN large=FALSE;
+	T_R2D_ARC_REGION rgn;
+	INT32 sa,sb,ea,eb;
+	INT32 a,b;
+
+	a=(br_x-ul_x)>>1;
+	b=(br_y-ul_y)>>1;
+
+	r2d_local_to_global(gc,&ul_x,&ul_y);
+	r2d_local_to_global(gc,&br_x,&br_y);
+
+
+	rgn.one_sector=0;
+	if (stop_angle-start_angle>180)
+		rgn.one_sector=1;
+
+	if (stop_angle-start_angle==360)
+         rgn.one_sector=2;
+
+	if (start_angle==stop_angle)
+		rgn.is_null=TRUE;
+	else
+		rgn.is_null=FALSE;
+	r2d_check_angles(&start_angle,&stop_angle);
+	
+
+	sa=a * r2d_cos(start_angle); // Q16
+	sb=  -b* r2d_sin(start_angle); // Q16
+	// Orthogonal is (sb,-sa)
+
+	ea= a * r2d_cos(stop_angle); // Q16
+	eb= -b* r2d_sin(stop_angle); // Q16
+	// Orthogonal is (-sb,sa)
+
+	//r2d_moveto(gc,ul_x+a,ul_y+b);
+	//r2d_lineto(gc,ul_x+a+(sa>>16),ul_y+b+(sb>>16));
+
+	// Orthogonal
+	//r2d_moveto(gc,ul_x+a,ul_y+b);
+	//r2d_lineto(gc,ul_x+a+(sb>>16),ul_y+b+((-sa)>>16));
+
+	//r2d_moveto(gc,ul_x+a,ul_y+b);
+	//r2d_lineto(gc,ul_x+a+(ea>>16),ul_y+b+(eb>>16));
+
+	rgn.sa=sa;
+	rgn.sb=sb;
+
+
+	rgn.ea=-ea;
+	rgn.eb=-eb;
+
+	
+
+#ifdef R2D_MIRRORED_X
+	   ul_x = r2d_get_width(gc)-1-ul_x;
+	   br_x = r2d_get_width(gc)-1-br_x;
+       R2D_SWAP(ul_x,br_x);
+
+	   r2d_arc_region_x_reflect(&rgn);
+#endif
+
+#ifdef R2D_MIRRORED_Y
+	   ul_y = r2d_get_height(gc)-1-ul_y;
+	   br_y = r2d_get_height(gc)-1-br_y;
+	   R2D_SWAP(ul_y,br_y);
+
+	   r2d_arc_region_y_reflect(&rgn);
+#endif
+
+   
+
+
+
+   r2d_df_arc(gc,ul_x,ul_y,br_x,br_y,&rgn,TRUE);
+   if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)gc)->p_frame_buffer)->kind==0)
+          r2d_update_region((T_R2D_GC*)gc,ul_x,ul_y,br_x,br_y);
+
+   r2d_check_and_send_event(gc);
+}
+
+///////////////////////////////////////
+//
+// Copy function
+//
+
+static void r2d_diagonal_mirror(T_R2D_SHAPE_PTR rectangle)
+{
+   T_R2D_RECTANGLE *r;
+   INT16 tmp;
+   r=(T_R2D_RECTANGLE *)rectangle;
+
+   tmp=r->ul_x;
+   r->ul_x=r->ul_y;
+   r->ul_y=tmp;
+
+   tmp=r->br_x;
+   r->br_x=r->br_y;
+   r->br_y=tmp;
+}
+
+// Global coordinates
+static void r2d_mirror_rectangle(T_R2D_GC_PTR gc,T_R2D_SHAPE_PTR rectangle)
+{
+   T_R2D_RECTANGLE *r;
+   //INT16 org_x,org_y;
+#ifdef R2D_MIRRORED_X
+   INT16 tmp;
+#endif
+   r=(T_R2D_RECTANGLE *)rectangle;
+
+   
+#ifdef R2D_MIRRORED_X
+   r->ul_x = r2d_get_width(gc)-r->ul_x;
+   r->br_x = r2d_get_width(gc)-r->br_x;
+   tmp=r->ul_x;
+	r->ul_x=r->br_x;
+	r->br_x=tmp;
+#endif
+
+#ifdef R2D_MIRRORED_Y
+	r->ul_y = r2d_get_height(gc)-r->ul_y;
+	r->br_y = r2d_get_height(gc)-r->br_y;
+
+    tmp=r->ul_y;
+	r->ul_y=r->br_y;
+	r->br_y=tmp;
+#endif
+
+	   
+}
+
+// Global coordinates
+// Here a rectangle with clipping convention is mirrored
+// The resulting rectangle (s_clipping) is used with drawing
+// convention. So, the boundary must be updated
+static void r2d_mirror_clip_rectangle(T_R2D_GC_PTR gc,T_R2D_SHAPE_PTR rectangle)
+{
+   T_R2D_RECTANGLE *r;
+//   INT16 org_x,org_y;
+#ifdef R2D_MIRRORED_X
+   INT16 tmp;
+#endif
+   r=(T_R2D_RECTANGLE *)rectangle;
+
+   
+   //Be careful br_x is NOT a line wich must be drawn
+#ifdef R2D_MIRRORED_X
+   r->ul_x = r2d_get_width(gc)-r->ul_x+1;
+   r->br_x = r2d_get_width(gc)-(r->br_x-1);
+   tmp=r->ul_x;
+	r->ul_x=r->br_x;
+	r->br_x=tmp;
+#endif
+
+#ifdef R2D_MIRRORED_Y
+	r->ul_y = r2d_get_height(gc)-r->ul_y+1;
+	r->br_y = r2d_get_height(gc)-(r->br_y-1);
+
+    tmp=r->ul_y;
+	r->ul_y=r->br_y;
+	r->br_y=tmp;
+#endif
+
+	   
+}
+
+// Compute the intersection of the clipping shape of srcGc with
+// the srcRectangle. Report he size reduction onto dstRectangle.
+// Compute the intersection between dstRectangle and the clipping rectangle
+// of dstGc and report the size reduction to srcRectangle.
+// At the end, one get two rectangles of same size.
+// The first one is contained in the clipping shape of srcGc
+// and the second one in the clipping shape of dstGc.
+// If the result is null, false is returned and no  copy is done
+// The rectangles are modified.
+static BOOLEAN r2d_clipped_is_not_null(T_R2D_GC* src_gc,T_R2D_GC *dst_gc,
+                                       T_R2D_RECTANGLE* src_rectangle,T_R2D_RECTANGLE* dst_rectangle)
+{
+  BOOLEAN result=TRUE;
+  //CC : clipping coordinate , RC : rectangle coordinate
+  INT16 delta,CC,RC;
+
+#if (R2D_DEBUG == R2D_ON)
+#if (R2D_DEBUG_WARNING == R2D_DEBUG_HIGH)
+	if (src_gc->p_clipping_shape==NULL)
+	  rvf_send_trace("R2D : clipping shape error",26, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, r2d_use_id );
+	if (dst_gc->p_clipping_shape==NULL)
+	  rvf_send_trace("R2D : clipping shape error",26, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, r2d_use_id );
+#endif
+#endif
+
+  //T_R2D_RECTANGLE *debug;
+
+  //debug=(T_R2D_RECTANGLE*)src_gc->p_clipping_shape;
+
+
+  // Intersections in src context
+  CC=r2d_get_xmin(src_gc->p_clipping_shape);
+  RC=r2d_get_xmin(src_rectangle);
+  if ( CC > RC)
+  {
+     delta=CC-RC;
+     src_rectangle->ul_x=CC;
+     dst_rectangle->ul_x+=delta;
+  }
+
+  CC=r2d_get_xmax(src_gc->p_clipping_shape);
+  RC=r2d_get_xmax(src_rectangle);
+  if ( RC > CC)
+  {
+     delta=RC-CC;
+     src_rectangle->br_x=CC;
+     dst_rectangle->br_x-=delta;
+  }
+
+  CC=r2d_get_ymin(src_gc->p_clipping_shape);
+  RC=r2d_get_ymin(src_rectangle);
+  if ( CC > RC)
+  {
+     delta=CC-RC;
+     src_rectangle->ul_y=CC;
+     dst_rectangle->ul_y+=delta;
+  }
+
+  CC=r2d_get_ymax(src_gc->p_clipping_shape);
+  RC=r2d_get_ymax(src_rectangle);
+  if ( RC > CC)
+  {
+     delta=RC-CC;
+     src_rectangle->br_y=CC;
+     dst_rectangle->br_y-=delta;
+  }
+
+  // Intersection in dst context
+  CC=r2d_get_xmin(dst_gc->p_clipping_shape);
+  RC=r2d_get_xmin(dst_rectangle);
+  if ( CC > RC)
+  {
+     delta=CC-RC;
+     dst_rectangle->ul_x=CC;
+     src_rectangle->ul_x+=delta;
+  }
+
+  CC=r2d_get_xmax(dst_gc->p_clipping_shape);
+  RC=r2d_get_xmax(dst_rectangle);
+  if ( RC > CC)
+  {
+     delta=RC-CC;
+     dst_rectangle->br_x=CC;
+     src_rectangle->br_x-=delta;
+  }
+
+  CC=r2d_get_ymin(dst_gc->p_clipping_shape);
+  RC=r2d_get_ymin(dst_rectangle);
+  if ( CC > RC)
+  {
+     delta=CC-RC;
+     dst_rectangle->ul_y=CC;
+     src_rectangle->ul_y+=delta;
+  }
+
+  CC=r2d_get_ymax(dst_gc->p_clipping_shape);
+  RC=r2d_get_ymax(dst_rectangle);
+  if ( RC > CC)
+  {
+     delta=RC-CC;
+     dst_rectangle->br_y=CC;
+     src_rectangle->br_y-=delta;
+  }
+
+  if (src_rectangle->ul_x>=src_rectangle->br_x)
+   result=FALSE;
+
+
+  if (src_rectangle->ul_y>=src_rectangle->br_y)
+   result=FALSE;
+
+  if (result)
+  {
+    /*printf("src: (%d,%d) -- (%d,%d)\n",src_rectangle->ul_x,src_rectangle->ul_y
+    ,src_rectangle->br_x,src_rectangle->br_y);
+    printf("dst: (%d,%d) -- (%d,%d)\n",dst_rectangle->ul_x,dst_rectangle->ul_y
+    ,dst_rectangle->br_x,dst_rectangle->br_y);*/
+
+    // Conversion of clipped rectangles to framebuffer coordinates
+    r2d_local_to_global(src_gc,&(src_rectangle->ul_x),&(src_rectangle->ul_y));
+    r2d_local_to_global(src_gc,&(src_rectangle->br_x),&(src_rectangle->br_y));
+    r2d_local_to_global(dst_gc,&(dst_rectangle->ul_x),&(dst_rectangle->ul_y));
+    r2d_local_to_global(dst_gc,&(dst_rectangle->br_x),&(dst_rectangle->br_y));
+
+#if (R2D_DEBUG == R2D_ON)
+#if (R2D_DEBUG_WARNING == R2D_DEBUG_HIGH)
+	if (src_rectangle->ul_x<0)
+	  rvf_send_trace("R2D : clipping shape error",26, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+
+	if (src_rectangle->ul_y<0)
+	  rvf_send_trace("R2D : clipping shape error",26, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+
+	if (dst_rectangle->ul_x<0)
+	  rvf_send_trace("R2D : clipping shape error",26, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+
+
+	if (dst_rectangle->ul_y<0)
+	  rvf_send_trace("R2D : clipping shape error",26, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+
+	if (src_rectangle->br_x>r2d_get_width(src_gc))
+	  rvf_send_trace("R2D : clipping shape error",26, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+
+	if (src_rectangle->br_y>r2d_get_height(src_gc))
+	  rvf_send_trace("R2D : clipping shape error",26, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+
+	if (dst_rectangle->br_x>r2d_get_width(dst_gc))
+	  rvf_send_trace("R2D : clipping shape error",26, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+
+	if (dst_rectangle->br_y>r2d_get_height(dst_gc))
+	  rvf_send_trace("R2D : clipping shape error",26, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+
+#endif
+#endif
+    /*printf("src: (%d,%d) -- (%d,%d)\n",src_rectangle->ul_x,src_rectangle->ul_y
+    ,src_rectangle->br_x,src_rectangle->br_y);
+    printf("dst: (%d,%d) -- (%d,%d)\n",dst_rectangle->ul_x,dst_rectangle->ul_y
+    ,dst_rectangle->br_x,dst_rectangle->br_y);*/
+  }
+
+  return(result);
+}
+
+T_R2D_ERROR           r2d_low_level_blit_rect(T_R2D_GC_PTR src_gc,T_R2D_GC_PTR dst_gc,
+                         T_R2D_SHAPE_PTR org_src_rectangle, T_R2D_SHAPE_PTR org_dst_rectangle,
+						 BOOLEAN src_mirrored,BOOLEAN dst_mirrored,
+						 BOOLEAN use_foreground_color)
+{
+
+   INT16 kind_src,kind_dst,org_bx,org_by,org_ux,org_uy;
+#if (R2D_ASM == R2D_ON)
+   T_R2D_DRAWING_MODE mode;
+#endif
+
+   T_R2D_SHAPE_PTR src_rectangle,dst_rectangle;
+
+#if (R2D_DEBUG == R2D_ON)
+#if (R2D_DEBUG_WARNING == R2D_DEBUG_HIGH)
+   if (src_gc==NULL)
+   {
+	  rvf_send_trace("src_gc NULL",11, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, r2d_use_id );
+	  rvf_delay(RVF_SECS_TO_TICKS(1));
+   }
+
+   if (dst_gc==NULL)
+   {
+	  rvf_send_trace("dst_gc NULL",11, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, r2d_use_id );
+	  rvf_delay(RVF_SECS_TO_TICKS(1));\
+   }
+
+   if (org_src_rectangle==NULL)
+   {
+	  rvf_send_trace("org_src NULL",12, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, r2d_use_id );
+	  rvf_delay(RVF_SECS_TO_TICKS(1));\
+   }
+
+   if (org_dst_rectangle==NULL)
+   {
+	  rvf_send_trace("org_dst NULL",12, NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, r2d_use_id );
+	  rvf_delay(RVF_SECS_TO_TICKS(1));\
+   }
+#endif
+#endif
+
+   
+
+    src_rectangle=r2d_clone_shape(r2d_mb_id,org_src_rectangle);
+    if (src_rectangle==NULL)
+	   return(R2D_MEMORY_ERR);
+	
+
+    dst_rectangle=r2d_clone_shape(r2d_mb_id,org_dst_rectangle);
+    if (dst_rectangle==NULL)
+    {
+	   r2d_release_shape(src_rectangle);
+	   return(R2D_MEMORY_ERR);
+    }
+
+	r2d_disable_refresh();
+
+	org_bx=((T_R2D_RECT*)src_rectangle)->br_x;
+    org_by=((T_R2D_RECT*)src_rectangle)->br_y;
+	org_ux=((T_R2D_RECT*)src_rectangle)->ul_x;
+    org_uy=((T_R2D_RECT*)src_rectangle)->ul_y;
+	r2d_local_to_global(src_gc,&org_ux,&org_uy);
+	r2d_local_to_global(src_gc,&org_bx,&org_by);
+    if (r2d_clipped_is_not_null(src_gc,dst_gc,src_rectangle,dst_rectangle))
+	{
+   // Rects are now clipped and in global coordinates
+
+if (src_mirrored) 
+  r2d_mirror_rectangle(src_gc,src_rectangle);
+
+if (dst_mirrored)
+   r2d_mirror_rectangle(dst_gc,dst_rectangle);
+
+// In that case we are drawing text.
+// If dst rect clipped by the right side, the src must be shifted to 
+// the start of the char
+// and must not remained aligned with the end of the char
+// (don't forget that since char is drawn with mirror its start
+// is corresponding to the right and bottom side of the bounding
+// rectangle)
+// There is the dual problem when the dst rect is clipped by the
+// left side
+
+if ((dst_mirrored) && (!src_mirrored))
+{
+  INT16 dbx,dby,dux,duy;
+  dbx=0;
+  dux=0;
+  dby=0;
+  duy=0;
+
+#ifdef R2D_MIRRORED_X
+  dbx=org_bx-((T_R2D_RECT*)src_rectangle)->br_x;
+  dux=org_ux-((T_R2D_RECT*)src_rectangle)->ul_x;
+#endif
+
+#ifdef R2D_MIRRORED_Y
+  dby=org_by-((T_R2D_RECT*)src_rectangle)->br_y;
+  duy=org_uy-((T_R2D_RECT*)src_rectangle)->ul_y;
+#endif
+
+  r2d_translate_shape(src_rectangle,dbx,dby);
+  r2d_translate_shape(src_rectangle,dux,duy);
+}
+
+
+
+ // After mirrong one is in L coordinates
+
+
+  if (((T_R2D_FRAMEBUFFER*)((T_R2D_GC*)dst_gc)->p_frame_buffer)->kind==0)
+          r2d_update_region((T_R2D_GC*)dst_gc,r2d_get_xmin(dst_rectangle),
+		  r2d_get_ymin(dst_rectangle),
+		  r2d_get_xmax(dst_rectangle),
+		  r2d_get_ymax(dst_rectangle));
+
+#if (R2D_REFRESH != R2D_VERTICAL)
+  r2d_diagonal_mirror(src_rectangle);
+  r2d_diagonal_mirror(dst_rectangle);
+#endif
+
+  // After swap one is in S coordinates
+
+   kind_src=((T_R2D_FRAMEBUFFER*)(((T_R2D_GC*)src_gc)->p_frame_buffer))->kind;
+   kind_dst=((T_R2D_FRAMEBUFFER*)(((T_R2D_GC*)dst_gc)->p_frame_buffer))->kind;
+
+    
+#if (R2D_ASM == R2D_OFF)
+
+   if ((kind_src!=R2D_FULL_KIND) && (kind_dst!=R2D_FULL_KIND))
+     IND_r2d_blit_lcd_to_lcd(src_gc,dst_gc,src_rectangle,dst_rectangle,use_foreground_color);
+   else
+   if ((kind_src==R2D_FULL_KIND) && (kind_dst==R2D_FULL_KIND))
+     IND_r2d_blit_color_to_color(src_gc,dst_gc,src_rectangle,dst_rectangle,use_foreground_color);
+   else
+   if ((kind_src!=R2D_FULL_KIND) && (kind_dst==R2D_FULL_KIND))
+     IND_r2d_blit_lcd_to_color(src_gc,dst_gc,src_rectangle,dst_rectangle,use_foreground_color);
+   else
+   if ((kind_src==R2D_FULL_KIND) && (kind_dst!=R2D_FULL_KIND))
+     IND_r2d_blit_color_to_lcd(src_gc,dst_gc,src_rectangle,dst_rectangle,use_foreground_color);
+
+#else
+   mode=(((T_R2D_GC*)dst_gc))->drawing_mode;
+   rvf_lock_mutex(r2d_g_blit_mutex);
+
+   if ((kind_src!=R2D_FULL_KIND) && (kind_dst!=R2D_FULL_KIND))
+   {
+	 r2d_patch_blit(dst_gc,mode,
+					use_foreground_color,R2D_LCDLCD);
+     r2d_blit_asm(src_gc,dst_gc,src_rectangle,dst_rectangle,
+	 use_foreground_color,R2D_LCDLCD);
+   }
+   else
+   if ((kind_src==R2D_FULL_KIND) && (kind_dst==R2D_FULL_KIND))
+   {
+	   r2d_patch_blit(dst_gc,mode,
+					use_foreground_color,R2D_COLORCOLOR);
+     r2d_blit_asm(src_gc,dst_gc,src_rectangle,dst_rectangle,
+	 use_foreground_color,R2D_COLORCOLOR);
+   }
+   else
+   if ((kind_src!=R2D_FULL_KIND) && (kind_dst==R2D_FULL_KIND))
+   {
+	   r2d_patch_blit(dst_gc,mode,
+					use_foreground_color,R2D_LCDCOLOR);
+     r2d_blit_asm(src_gc,dst_gc,src_rectangle,dst_rectangle,
+	 use_foreground_color,R2D_LCDCOLOR);
+   }
+   else
+   if ((kind_src==R2D_FULL_KIND) && (kind_dst!=R2D_FULL_KIND))
+   {
+	   r2d_patch_blit(dst_gc,mode,
+					use_foreground_color,R2D_COLORLCD);
+     r2d_blit_asm(src_gc,dst_gc,src_rectangle,dst_rectangle,
+	 use_foreground_color,R2D_COLORLCD);
+   }
+   rvf_unlock_mutex(r2d_g_blit_mutex);
+
+#endif
+
+   
+   r2d_check_and_send_event(dst_gc);
+	}
+	 r2d_release_shape(src_rectangle);
+   r2d_release_shape(dst_rectangle);
+   r2d_enable_refresh();
+   return(R2D_OK);
+}
+
+T_R2D_ERROR           r2d_blit_rect(T_R2D_GC_PTR src_gc,T_R2D_GC_PTR dst_gc,
+                         T_R2D_SHAPE_PTR org_src_rectangle, 
+						 T_R2D_SHAPE_PTR org_dst_rectangle,
+						 BOOLEAN use_foreground_color)
+{
+		// Allow reversing if needed
+	return(r2d_low_level_blit_rect(src_gc,dst_gc,org_src_rectangle,org_dst_rectangle,TRUE,TRUE,use_foreground_color));
+}
+
+
+
+///////////////////////////////////////
+//
+// Text functions
+//
+
+T_R2D_CHAR_METRIC_PTR r2d_get_char_metrics(T_R2D_GC_PTR gc,UINT32 the_char)
+{
+  INT32 *p;
+  INT32 new_char;
+
+  new_char=r2d_ptree_find(((T_R2D_GC*)gc)->font_table,the_char);
+
+  p=((T_R2D_GC*)gc)->font_metrics;
+  p+=(new_char<<3);
+
+  *p++;
+
+  return((T_R2D_CHAR_METRIC_PTR)p);
+}
+
+T_R2D_ERROR r2d_draw_char(T_R2D_GC_PTR font_cache_gc,
+                    T_R2D_GC_PTR gc,
+                    T_R2D_GC_PTR fontgc,
+                    INT16 x,INT16 y,INT16 org_size,T_R2D_CHAR_METRIC_PTR met)
+{
+    
+    T_R2D_SHAPE_PTR src_rectangle,dst_rectangle,tdst_rectangle;
+    
+
+    INT32 *p;
+    INT16 underline;
+//	INT32 new_char;
+	INT16 style;
+	INT16 ascent;
+	T_R2D_ERROR err;
+
+	style=((T_R2D_GC *)gc)->internal_text_style;
+	ascent=((T_R2D_GC*)gc)->font_metrics[R2D_BUFFER_WIDTH_POS+1];
+
+	err=R2D_OK;
+
+
+    p=((T_R2D_GC*)gc)->font_metrics;
+    underline=*--p;
+   
+
+    
+    {
+       
+	   
+       
+       if (style & R2D_BOLD)
+       {
+//           T_R2D_DRAWING_MODE the_mode;
+//           T_R2D_ARGB_COLOR bg_color;
+
+#if (R2D_REFRESH == R2D_VERTICAL)
+           src_rectangle=r2d_new_rectangle(r2d_mb_id,r2d_get_char_x(met),0,
+		   r2d_get_char_x(met)+r2d_get_char_width(met),
+		   r2d_get_char_height(met));
+#else
+		   // In horizontal mode char are organized vertically
+		   src_rectangle=r2d_new_rectangle(r2d_mb_id,0,r2d_get_char_x(met),
+			   r2d_get_char_width(met),r2d_get_char_x(met)+r2d_get_char_height(met));
+#endif
+		   if (src_rectangle==NULL)
+		   {
+			   err=R2D_MEMORY_ERR;
+			   return(err);
+		   }
+		   //dst_rectangle=r2d_new_rectangle(0,0,char_width+1,char_height);
+		   //if (dst_rectangle==NULL)
+		   //{
+		//	   r2d_release_shape(src_rectangle);
+		//	   *err=1;
+		//	   return(0);
+		   //}
+           
+           // erase temporary char if R2D is starting to draw a new char
+           // (it is not always the case with diacritics chars)
+           
+		   //r2d_set_background_color_with_argb(font_cache_gc,0,0,255,0);
+		   r2d_fill_rectangle(font_cache_gc,-1,-1,r2d_get_width(font_cache_gc)+1,
+			   r2d_get_height(font_cache_gc)+1);
+		   //r2d_set_background_color_with_argb(font_cache_gc,0,255,255,255);
+
+           //r2d_release_shape(dst_rectangle);
+                
+           // Create destination rectangles 
+		   dst_rectangle=r2d_new_rectangle(r2d_mb_id,0,0,r2d_get_char_width(met),
+			   r2d_get_char_height(met));
+		   if (dst_rectangle==NULL)
+		   {
+			   r2d_release_shape(src_rectangle);
+			   err=R2D_MEMORY_ERR;
+			   return(err);
+		   }
+
+           tdst_rectangle=r2d_clone_shape(r2d_mb_id,dst_rectangle);
+		   if (tdst_rectangle==NULL)
+		   {
+			   r2d_release_shape(src_rectangle);
+			   r2d_release_shape(dst_rectangle);
+			   err=R2D_MEMORY_ERR;
+			   return(err);
+		   }
+
+           r2d_translate_shape(tdst_rectangle,1,0);
+
+
+           // set mode
+           r2d_set_drawing_mode(font_cache_gc,R2D_OR_MODE);
+
+           // Build bold char
+
+           if (r2d_low_level_blit_rect(fontgc,font_cache_gc,src_rectangle,dst_rectangle,
+			   FALSE,FALSE,FALSE)!=R2D_OK)
+		   {
+			   r2d_release_shape(src_rectangle);
+			   r2d_release_shape(dst_rectangle);
+			   r2d_release_shape(tdst_rectangle);
+			   err=R2D_MEMORY_ERR;
+			   return(err);
+		   }
+
+           if (r2d_low_level_blit_rect(fontgc,font_cache_gc,src_rectangle,tdst_rectangle,
+			   FALSE,FALSE,FALSE)!=R2D_OK)
+		   {
+			   r2d_release_shape(src_rectangle);
+			   r2d_release_shape(dst_rectangle);
+			   r2d_release_shape(tdst_rectangle);
+			   err=R2D_MEMORY_ERR;
+			   return(err);
+		   }
+
+		   if (style & R2D_STRIKETHROUGH)
+		   {
+		     r2d_moveto(font_cache_gc,0,ascent>>1);
+		     r2d_lineto(font_cache_gc,r2d_get_char_width(met),ascent>>1);
+		   }
+
+           // Reset mode
+           r2d_set_drawing_mode(font_cache_gc,R2D_COPY_MODE);
+
+           r2d_release_shape(tdst_rectangle);
+
+           r2d_release_shape(dst_rectangle);
+           r2d_release_shape(src_rectangle);
+
+           // Recreate src and dest rectangle
+           src_rectangle=r2d_new_rectangle(r2d_mb_id,0,0,r2d_get_char_width(met)+1,
+			   r2d_get_char_height(met));
+		   if (src_rectangle==NULL)
+		   {
+			   err=R2D_MEMORY_ERR;
+			   return(err);
+		   }
+		   
+		   if (org_size != 0)
+		  {
+			  INT16 tmp;
+
+			  tmp=(org_size-r2d_get_char_width(met))>>1;
+			  dst_rectangle=r2d_new_rectangle(r2d_mb_id,x+tmp,y-r2d_get_char_height(met),
+				  x+tmp+r2d_get_char_width(met)+1,y);
+		  }
+		  else
+          dst_rectangle=r2d_new_rectangle(r2d_mb_id,x+r2d_get_char_org_x(met),
+		  y-r2d_get_char_org_y(met),
+		  x+r2d_get_char_width(met)+r2d_get_char_org_x(met)+1,
+		  y+r2d_get_char_height(met)-r2d_get_char_org_y(met));
+          
+           if (dst_rectangle==NULL)
+		   {
+			   r2d_release_shape(src_rectangle);
+			   err=R2D_MEMORY_ERR;
+			   return(err);
+		   }
+           //printf("free_pos=%d, char_width=%d\n",free_pos,char_width);
+           if (r2d_low_level_blit_rect(font_cache_gc,gc,src_rectangle,dst_rectangle,
+			   FALSE,TRUE,TRUE)!=R2D_OK)
+		   {
+			   r2d_release_shape(src_rectangle);
+			   r2d_release_shape(dst_rectangle);
+			   err=R2D_MEMORY_ERR;
+			   return(err);
+		   }
+
+       }
+       else 
+       {
+          #if (R2D_REFRESH == R2D_VERTICAL)
+           src_rectangle=r2d_new_rectangle(r2d_mb_id,r2d_get_char_x(met),
+			   0,
+			   r2d_get_char_x(met)+r2d_get_char_width(met),
+			   r2d_get_char_height(met));
+#else
+		   // In horizontal mode char are organized vertically
+		   src_rectangle=r2d_new_rectangle(r2d_mb_id,0,
+			   r2d_get_char_x(met),
+			   r2d_get_char_width(met),
+			   r2d_get_char_x(met)+r2d_get_char_height(met));
+#endif
+		  if (src_rectangle==NULL)
+		  {
+			   err=R2D_MEMORY_ERR;
+			   return(err);
+		  }
+          
+		  if (org_size != 0)
+		  {
+			  INT16 tmp;
+
+			  tmp=(org_size-r2d_get_char_width(met))>>1;
+			  dst_rectangle=r2d_new_rectangle(r2d_mb_id,x+tmp,
+				  y-r2d_get_char_height(met),
+				  x+tmp+r2d_get_char_width(met),y);
+		  }
+		  else
+          dst_rectangle=r2d_new_rectangle(r2d_mb_id,
+		  x+r2d_get_char_org_x(met),
+		  y-r2d_get_char_org_y(met),
+		  x+r2d_get_char_width(met)+r2d_get_char_org_x(met),
+		  y+r2d_get_char_height(met)-r2d_get_char_org_y(met));
+          
+		  if (dst_rectangle==NULL)
+		  {
+			   r2d_release_shape(src_rectangle);
+			   err=R2D_MEMORY_ERR;
+			   return(err);
+		  }
+
+		  if (style & R2D_STRIKETHROUGH)
+		  {
+			  r2d_fill_rectangle(font_cache_gc,-1,-1,r2d_get_width(font_cache_gc)+1,
+			   r2d_get_height(font_cache_gc)+1);
+			  tdst_rectangle=r2d_new_rectangle(r2d_mb_id,0,0,r2d_get_char_width(met),
+			   r2d_get_char_height(met));
+
+            if (tdst_rectangle==NULL)
+			{
+			   r2d_release_shape(src_rectangle);
+			   r2d_release_shape(dst_rectangle);
+			   err=R2D_MEMORY_ERR;
+			   return(err);
+			}
+
+              err=r2d_low_level_blit_rect(fontgc,font_cache_gc,src_rectangle,tdst_rectangle,
+			  FALSE,FALSE,FALSE);
+			  if (err==R2D_OK)
+			  {
+			    r2d_moveto(font_cache_gc,0,ascent>>1);
+			    r2d_lineto(font_cache_gc,r2d_get_char_width(met),ascent>>1);
+                err=r2d_low_level_blit_rect(font_cache_gc,gc,tdst_rectangle,dst_rectangle,
+			      FALSE,TRUE,TRUE);
+			  }
+
+			  r2d_release_shape(tdst_rectangle);
+		  }
+		  else
+		  {
+			  err=r2d_low_level_blit_rect(fontgc,gc,src_rectangle,dst_rectangle,
+			  FALSE,TRUE,TRUE);
+		  }
+
+          if (err!=R2D_OK)
+		  {
+			   r2d_release_shape(src_rectangle);
+			   r2d_release_shape(dst_rectangle);
+			   err=R2D_MEMORY_ERR;
+			   return(err);
+		  }
+
+
+       }
+
+
+
+       if (style & R2D_UNDERLINED)
+       {
+           r2d_moveto(gc,x+r2d_get_char_org_x(met),y+underline);
+           r2d_lineto(gc,x+r2d_get_char_org_x(met)+
+			   r2d_get_char_dx(met),y+underline);
+       }
+       
+       r2d_release_shape(src_rectangle);
+       r2d_release_shape(dst_rectangle);
+
+       
+    }
+   return(err);
+}
+
+
+#define R2D_ENDIAN_SWAP(a) {misc=(a&0x0FF) << 8;a=a>>8;a=a|misc;a=a&0x0FFFF;}
+
+UINT32 r2d_get_next_char(T_R2D_UTF16 *the_text,INT16 *pos,UINT16 max_chars,
+							  BOOLEAN *swapping)
+{
+	T_R2D_UTF16 current;
+	INT16 temp;
+	INT32 the_char;
+	BOOLEAN must_display,combining,done;
+	T_R2D_UTF16 next, misc;
+//	T_R2D_CHAR_METRIC_PTR p;
+	// Swapping should be set to FALSE first time this function is called
+
+
+	
+	
+	// current char consider a WHOLE char
+	// (when combining char are used, they are considered
+	// as making ONE char)
+	done=FALSE;
+
+   // Detection of byte order mark
+	if (the_text[*pos] == 0xFFFE)
+	{
+		*swapping=TRUE;
+		*pos=*pos+1;
+	}
+    while ((!done) && (the_text[*pos]!=0) && (*pos<max_chars))
+    {
+		must_display=FALSE;
+		combining=FALSE;
+
+        current=the_text[*pos];
+		*pos=*pos+1;
+        if (*swapping)
+			R2D_ENDIAN_SWAP(current);
+		
+	    if ((current != 0x0FFFF))
+		{
+	     // If High-surrogate 
+		 if ((current >= 0xd800) && (current <= 0xdbff))
+		 {
+			 // Then check that next char if a low-surrogate
+			 next=the_text[*pos];
+			 if (*swapping)
+			   R2D_ENDIAN_SWAP(next);
+			 
+			 if ((next >= 0xdc00) && (next<=0xdfff))
+			 {
+				 *pos=*pos+1;
+				 the_char=(current - 0xd800) * 0x400 + (next - 0xdc00) + 0x10000;
+				 must_display=TRUE;
+			 }
+			 else
+			 {
+			     rvf_send_trace("r2d_get_next_char() missing low surrogate char",
+			        strlen("r2d_get_next_char() missing low surrogate char"), NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+			 }
+		 }
+		 else
+		 {
+			 // If not a high-surrogate then check it is not
+			 // a low-surrogate
+			 if ((current < 0xdc00) || (current>0xdfff))
+			 {
+				 the_char=current;
+				 must_display=TRUE;
+			 }
+			 else
+			 {
+			     rvf_send_trace("R2D : r2d_get_next_char unexpected low surrogate char",
+			        strlen("R2D : r2d_get_next_char unexpected low surrogate char"), NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+			 }
+
+		 }
+		}
+		
+		 if ((the_char>=0x300) && (the_char <=0x36f))
+			combining=TRUE;
+
+		 if (must_display)
+		 {
+		   
+		   if (!combining)
+		   {
+		     // Next char
+			   done=TRUE;
+			   
+		   }
+		 }
+		
+       
+    }
+	if (done)
+	  return(the_char);
+	else
+		return(0);
+}
+//
+static T_R2D_ERROR          r2d_low_draw_text(T_R2D_GC_PTR gc,INT16 x,INT16 y,
+									   T_R2D_UTF16 *the_text,
+									   T_R2D_TEXT_DRAWING_SETTINGS settings,
+									   UINT16 max_words,
+									   UINT16 *width,
+									   UINT16 *height)
+{
+    T_R2D_GC_PTR fontgc;
+    T_R2D_GC_PTR cache_gc;
+    BOOLEAN draw;
+    T_R2D_DRAWING_MODE the_mode;
+	T_R2D_UTF16 *temp;
+	INT16 err;
+	INT16 current_word;
+	INT32 the_char;
+	BOOLEAN must_display,swapping,combining;
+	INT16 delta,deltay,misc,current_y,maxx,delta_line,y_org;
+	T_R2D_UTF16 next,current;
+	T_R2D_CHAR_METRIC_PTR p;
+	
+	
+    INT16 x_org;
+
+
+
+	draw=((settings & R2D_TEXT_DRAW)==R2D_TEXT_DRAW);
+
+	if (the_text==NULL)
+		return(R2D_MEMORY_ERR);
+
+	x_org=x;
+	y_org=y;
+
+	maxx=0;
+
+	
+
+	current_word=0; // We are at first word in string
+	// current word consider a word which may be a partial char
+	// (when combining char are used, they are considered
+	// as making ONE char)
+    swapping=FALSE;
+
+    if (draw)
+    {
+        // +1 used since blit_rect is using a bounding rect
+        cache_gc=r2d_new_font_buffer_context(r2d_mb_id,((T_R2D_GC*)gc)->font_frame_buffer->kind,
+            ((T_R2D_GC*)gc)->font_metrics[R2D_BUFFER_WIDTH_POS]+1,// max width for free area
+            ((T_R2D_GC*)gc)->font_metrics[R2D_BUFFER_WIDTH_POS+1]+ // ascent
+            ((T_R2D_GC*)gc)->font_metrics[R2D_BUFFER_WIDTH_POS+2]+ // descent
+            ((T_R2D_GC*)gc)->font_metrics[R2D_BUFFER_WIDTH_POS+3]); // leading
+        
+        delta_line=((T_R2D_GC*)gc)->font_metrics[R2D_BUFFER_WIDTH_POS+1]+ // ascent
+            ((T_R2D_GC*)gc)->font_metrics[R2D_BUFFER_WIDTH_POS+2]+ // descent
+            ((T_R2D_GC*)gc)->font_metrics[R2D_BUFFER_WIDTH_POS+3];
+        
+        
+        if (cache_gc==NULL)
+            return(R2D_MEMORY_ERR);
+        
+        r2d_context_lock(cache_gc);
+        
+        fontgc=r2d_new_context(r2d_mb_id,((T_R2D_GC*)gc)->font_frame_buffer);
+        //r2d_release_framebuffer(((T_R2D_GC*)gc)->font_frame_buffer);
+        
+        
+        if (fontgc==NULL)
+        {
+            r2d_context_unlock(cache_gc);
+            r2d_release_context(cache_gc);
+            return(R2D_MEMORY_ERR);
+        }
+        
+        r2d_context_lock(fontgc);
+    
+        r2d_set_background_color_with_argb(fontgc,0,255,255,255);
+        r2d_set_foreground_color_with_argb(fontgc,0,0,0,0);
+    
+        the_mode=r2d_get_drawing_mode(gc);
+    }
+    
+    err=R2D_OK;
+	delta=0;
+	deltay=0;
+	current_y=y;
+	// Detection of byte order mark
+	if (*the_text == 0xFFFE)
+	{
+		swapping=TRUE;
+		the_text++;
+		current_word++;
+	}
+    while((*the_text!=0) && (err==R2D_OK) && (current_word<max_words))
+    {
+		must_display=FALSE;
+		combining=FALSE;
+		if (the_char!=0xa)
+		  x+=delta;
+
+		delta = 0;
+
+        current=*the_text++;
+		current_word++;
+
+		if (swapping)
+			R2D_ENDIAN_SWAP(current);
+		
+	    if ((current != 0x0FFFF))
+		{
+	     // If High-surrogate 
+		 if ((current >= 0xd800) && (current <= 0xdbff))
+		 {
+			 // Then check that next char if a low-surrogate
+			 next=*the_text;
+			 if (swapping)
+			    R2D_ENDIAN_SWAP(next);
+			 if ((next >= 0xdc00) && (next<=0xdfff))
+			 {
+				 the_text++;
+				 current_word++;
+				 the_char=(current - 0xd800) * 0x400 + (next - 0xdc00) + 0x10000;
+				 must_display=TRUE;
+			 }
+			 else
+			 {
+			     rvf_send_trace("r2d_low_draw_text() missing low surrogate char",
+			        strlen("r2d_low_draw_text() missing low surrogate char"), NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+			 }
+		 }
+		 else
+		 {
+			 // If not a high-surrogate then check it is not
+			 // a low-surrogate
+			 if ((current < 0xdc00) || (current>0xdfff))
+			 {
+				 the_char=current;
+				 must_display=TRUE;
+			 }
+			 else
+			 {
+			     rvf_send_trace("r2d_low_draw_text() unexpected low surrogate char",
+			        strlen("r2d_low_draw_text() unexpected low surrogate char"), NULL_PARAM, 
+				   RV_TRACE_LEVEL_ERROR, R2D_USE_ID );
+			 }
+		 }
+		}
+		
+		 if ((the_char>=0x300) && (the_char <=0x36f))
+			combining=TRUE;
+		 else
+			current_y=y;
+
+		 if (the_char==0xa)
+		 {
+			 if ((x-x_org)>maxx)
+				 maxx=x-x_org;
+			 
+			 if ((x_org-x)>maxx)
+				 maxx=x_org-x;
+			 x=x_org;
+			 y += delta_line;
+
+		 }
+
+		 if ((must_display) && (the_char!=0xa))
+		 {
+		   p=r2d_get_char_metrics(gc,the_char);
+		   if (combining)
+		   {
+			  
+			  if (draw)
+			  {
+				
+				current_y -= deltay;
+				x=x-delta;
+				
+				if (((T_R2D_GC*)gc)->script_mode==R2D_LEFT_TO_RIGHT)
+                  err=r2d_draw_char(cache_gc,gc,fontgc,x,current_y,delta,p);
+				else
+                  err=r2d_draw_char(cache_gc,gc,fontgc,x+delta,current_y,delta,p);
+				deltay=r2d_get_char_height(p);
+			  }
+		   }
+		   else
+		   {
+		     if (draw)
+			 {
+				 
+                
+				if (((T_R2D_GC*)gc)->script_mode==R2D_LEFT_TO_RIGHT)
+				{
+				  delta=r2d_get_char_dx(p);
+				  err=r2d_draw_char(cache_gc,gc,fontgc,x,current_y,0,p);
+			      
+				}
+				else
+				{
+				   delta=-r2d_get_char_dx(p);
+				   err=r2d_draw_char(cache_gc,gc,fontgc,x+delta,current_y,0,p);
+				}
+				deltay=r2d_get_char_height(p);
+				
+			 }
+			 else
+			 {
+				 
+				if (((T_R2D_GC*)gc)->script_mode==R2D_LEFT_TO_RIGHT)
+			      delta=r2d_get_char_dx(p);
+				else
+                  delta=-r2d_get_char_dx(p);
+			 }
+		   }
+		 }
+		
+       
+    }
+	if (the_char!=0xa)
+	  x+=delta;
+
+	if ((x-x_org)>maxx)
+				 maxx=x-x_org;
+			 
+	if ((x_org-x)>maxx)
+				 maxx=x_org-x;
+
+	/*if (((T_R2D_GC*)gc)->script_mode==R2D_LEFT_TO_RIGHT)
+	  *result=x-x_org;
+	else
+      *result=x_org-x;*/
+
+	*width=maxx;
+	*height=y-y_org+delta_line;
+
+    if (draw)
+    {
+        r2d_set_drawing_mode(gc,the_mode);
+
+        r2d_context_unlock(fontgc);
+        r2d_context_unlock(cache_gc);
+        r2d_release_context(fontgc);
+        r2d_release_context(cache_gc);
+
+	    r2d_check_and_send_event(gc);
+    }
+
+	return(err);
+	
+}
+
+T_R2D_ERROR   r2d_get_text_width(T_R2D_GC_PTR gc,T_R2D_UTF16 *the_text,UINT16 *size)
+{
+    UINT16 x,y;
+	T_R2D_ERROR err;
+	err=R2D_OK;
+
+
+    err=r2d_low_draw_text(gc,0,0,the_text,0,-1,&x,&y);
+
+	if (err!=R2D_OK)
+	 *size=1;
+    else
+	{
+	  if (x>=0)
+        *size=x;
+      else
+        *size=-x;
+	}
+
+	return(err);
+}
+
+T_R2D_ERROR   r2d_get_width_of_chars(T_R2D_GC_PTR gc,T_R2D_UTF16 *the_text,
+									 UINT16 nb_chars,UINT16 *size)
+{
+    UINT16 x,y;
+	T_R2D_ERROR err;
+
+    err=r2d_low_draw_text(gc,0,0,the_text,0,nb_chars,&x,&y);
+
+
+	if (err!=R2D_OK)
+		*size=1;
+    else
+	{
+	  if (x>=0)
+        *size=x;
+      else
+        *size=-x;
+	}
+
+	return(err);
+}
+
+
+T_R2D_ERROR          r2d_draw_text(T_R2D_GC_PTR gc,INT16 x,INT16 y,T_R2D_UTF16 *the_text)
+{
+	UINT16 r,h;
+	return(r2d_low_draw_text(gc,x,y,the_text,R2D_TEXT_DRAW,-1,&r,&h));
+}
+
+
+T_R2D_ERROR          r2d_draw_chars(T_R2D_GC_PTR gc,INT16 x,INT16 y,
+									T_R2D_UTF16 *the_text,UINT16 nb_chars)
+{
+	UINT16 r,h;
+	return(r2d_low_draw_text(gc,x,y,the_text,R2D_TEXT_DRAW,nb_chars,&r,&h));
+}
+
+T_R2D_ERROR   r2d_draw_chars_and_get_size(T_R2D_GC_PTR gc,INT16 x,INT16 y,T_R2D_UTF16 *the_text,
+							 UINT16 nb_words16,UINT16 *width,UINT16 *height)
+{
+	return(r2d_low_draw_text(gc,x,y,the_text,R2D_TEXT_DRAW,nb_words16,width,height));
+}
+
+T_R2D_UTF16 *r2d_new_unicode_from_cstring(T_RVF_MB_ID bank,unsigned char *the_string)
+{
+ T_R2D_UTF16 *result,*p;
+// INT16 i;
+ INT16 index;
+
+ R2D_MALLOC(bank,T_R2D_UTF16,sizeof(T_R2D_UTF16)*(strlen((char*)the_string)+1),result);
+ if (result!=NULL)
+ {
+   p=result;
+   while(*the_string!='\0')
+   {
+	 index=*the_string++;
+     *p++ = R2D_WIN_LATIN1_TO_UNICODE[index];
+   }
+   *p++ = 0;
+ }
+ return(result);
+}
+
+T_R2D_UTF16 *r2d_new_unicode_from_pstring(T_RVF_MB_ID bank,unsigned char *the_string)
+{
+ T_R2D_UTF16 *result,*p;
+ INT16 l,index;
+// INT16 i;
+
+ R2D_MALLOC(bank,T_R2D_UTF16,sizeof(T_R2D_UTF16)*(the_string[0]+1),result);
+ if (result!=NULL)
+ {
+   p=result;
+   l=*the_string++;
+
+   while(l!=0)
+   {
+	 index=*the_string++;
+     *p++ = R2D_WIN_LATIN1_TO_UNICODE[index];
+     l--;
+   }
+   *p++ = 0;
+ }
+ return(result);
+}
+
+
+void          r2d_get_font_info(T_R2D_GC_PTR gc,INT16 *ascent,INT16 *descent,INT16 *leading)
+{
+   INT32 *p;
+
+    
+   p=((T_R2D_GC*)gc)->font_metrics;
+
+   *p--;
+   *leading=*--p;
+   *descent=*--p;
+   *ascent=*--p;
+   
+}
+
+INT16 r2d_str_nb_word16(T_R2D_UTF16 *l)
+{
+	INT16 res;
+	T_R2D_UTF16 *p;
+
+	if (l==NULL)
+		return 0;
+
+	p=l;
+	res=0;
+
+	while(*p!=0)
+	{
+		p++;
+		res++;
+	}
+	return(res);
+}
+
+T_R2D_UTF16 *r2d_duplicate_text(T_RVF_MB_ID bank,T_R2D_UTF16 *l)
+{
+	T_R2D_UTF16 *ret;
+	INT16 len;
+
+	if (l==NULL)
+		return(NULL);
+
+	len=r2d_str_nb_word16(l);
+	rvf_get_buf(bank,(len+1)*sizeof(T_R2D_UTF16),(void*)&ret);
+	if (ret)
+	{
+		memcpy(ret,l,(len+1)*2);
+		return(ret);
+	}
+    else return(NULL);
+}
+
+
+
+T_R2D_SCRIPT_MODE r2d_get_script_mode(T_R2D_GC_PTR gc)
+{
+	return(((T_R2D_GC*)gc)->script_mode);
+}
+
+void r2d_set_script_mode(T_R2D_GC_PTR gc,T_R2D_SCRIPT_MODE mode)
+{
+	((T_R2D_GC*)gc)->script_mode=mode;
+}