Blame gdk/x11/gdkdrawable-x11.c

Packit 98cdb6
/* GIMP Drawing Kit
Packit 98cdb6
 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
Packit 98cdb6
 *
Packit 98cdb6
 * This library is free software; you can redistribute it and/or
Packit 98cdb6
 * modify it under the terms of the GNU Lesser General Public
Packit 98cdb6
 * License as published by the Free Software Foundation; either
Packit 98cdb6
 * version 2 of the License, or (at your option) any later version.
Packit 98cdb6
 *
Packit 98cdb6
 * This library is distributed in the hope that it will be useful,
Packit 98cdb6
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 98cdb6
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 98cdb6
 * Lesser General Public License for more details.
Packit 98cdb6
 *
Packit 98cdb6
 * You should have received a copy of the GNU Lesser General Public
Packit 98cdb6
 * License along with this library; if not, write to the
Packit 98cdb6
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Packit 98cdb6
 * Boston, MA 02111-1307, USA.
Packit 98cdb6
 */
Packit 98cdb6
Packit 98cdb6
/*
Packit 98cdb6
 * Modified by the GTK+ Team and others 1997-2000.  See the AUTHORS
Packit 98cdb6
 * file for a list of people on the GTK+ Team.  See the ChangeLog
Packit 98cdb6
 * files for a list of changes.  These files are distributed with
Packit 98cdb6
 * GTK+ at ftp://ftp.gtk.org/pub/gtk/. 
Packit 98cdb6
 */
Packit 98cdb6
Packit 98cdb6
#include "config.h"
Packit 98cdb6
Packit 98cdb6
#include "gdkx.h"
Packit 98cdb6
#include "gdkregion-generic.h"
Packit 98cdb6
Packit 98cdb6
#include <cairo-xlib.h>
Packit 98cdb6
Packit 98cdb6
#include <stdlib.h>
Packit 98cdb6
#include <string.h>		/* for memcpy() */
Packit 98cdb6
Packit 98cdb6
#if defined (HAVE_IPC_H) && defined (HAVE_SHM_H) && defined (HAVE_XSHM_H)
Packit 98cdb6
#define USE_SHM
Packit 98cdb6
#endif
Packit 98cdb6
Packit 98cdb6
#ifdef USE_SHM
Packit 98cdb6
#include <X11/extensions/XShm.h>
Packit 98cdb6
#endif /* USE_SHM */
Packit 98cdb6
Packit 98cdb6
#include "gdkprivate-x11.h"
Packit 98cdb6
#include "gdkdrawable-x11.h"
Packit 98cdb6
#include "gdkpixmap-x11.h"
Packit 98cdb6
#include "gdkscreen-x11.h"
Packit 98cdb6
#include "gdkdisplay-x11.h"
Packit 98cdb6
Packit 98cdb6
#include "gdkalias.h"
Packit 98cdb6
Packit 98cdb6
static void gdk_x11_draw_rectangle (GdkDrawable    *drawable,
Packit 98cdb6
				    GdkGC          *gc,
Packit 98cdb6
				    gboolean        filled,
Packit 98cdb6
				    gint            x,
Packit 98cdb6
				    gint            y,
Packit 98cdb6
				    gint            width,
Packit 98cdb6
				    gint            height);
Packit 98cdb6
static void gdk_x11_draw_arc       (GdkDrawable    *drawable,
Packit 98cdb6
				    GdkGC          *gc,
Packit 98cdb6
				    gboolean        filled,
Packit 98cdb6
				    gint            x,
Packit 98cdb6
				    gint            y,
Packit 98cdb6
				    gint            width,
Packit 98cdb6
				    gint            height,
Packit 98cdb6
				    gint            angle1,
Packit 98cdb6
				    gint            angle2);
Packit 98cdb6
static void gdk_x11_draw_polygon   (GdkDrawable    *drawable,
Packit 98cdb6
				    GdkGC          *gc,
Packit 98cdb6
				    gboolean        filled,
Packit 98cdb6
				    GdkPoint       *points,
Packit 98cdb6
				    gint            npoints);
Packit 98cdb6
static void gdk_x11_draw_text      (GdkDrawable    *drawable,
Packit 98cdb6
				    GdkFont        *font,
Packit 98cdb6
				    GdkGC          *gc,
Packit 98cdb6
				    gint            x,
Packit 98cdb6
				    gint            y,
Packit 98cdb6
				    const gchar    *text,
Packit 98cdb6
				    gint            text_length);
Packit 98cdb6
static void gdk_x11_draw_text_wc   (GdkDrawable    *drawable,
Packit 98cdb6
				    GdkFont        *font,
Packit 98cdb6
				    GdkGC          *gc,
Packit 98cdb6
				    gint            x,
Packit 98cdb6
				    gint            y,
Packit 98cdb6
				    const GdkWChar *text,
Packit 98cdb6
				    gint            text_length);
Packit 98cdb6
static void gdk_x11_draw_drawable  (GdkDrawable    *drawable,
Packit 98cdb6
				    GdkGC          *gc,
Packit 98cdb6
				    GdkPixmap      *src,
Packit 98cdb6
				    gint            xsrc,
Packit 98cdb6
				    gint            ysrc,
Packit 98cdb6
				    gint            xdest,
Packit 98cdb6
				    gint            ydest,
Packit 98cdb6
				    gint            width,
Packit 98cdb6
				    gint            height,
Packit 98cdb6
				    GdkDrawable    *original_src);
Packit 98cdb6
static void gdk_x11_draw_points    (GdkDrawable    *drawable,
Packit 98cdb6
				    GdkGC          *gc,
Packit 98cdb6
				    GdkPoint       *points,
Packit 98cdb6
				    gint            npoints);
Packit 98cdb6
static void gdk_x11_draw_segments  (GdkDrawable    *drawable,
Packit 98cdb6
				    GdkGC          *gc,
Packit 98cdb6
				    GdkSegment     *segs,
Packit 98cdb6
				    gint            nsegs);
Packit 98cdb6
static void gdk_x11_draw_lines     (GdkDrawable    *drawable,
Packit 98cdb6
				    GdkGC          *gc,
Packit 98cdb6
				    GdkPoint       *points,
Packit 98cdb6
				    gint            npoints);
Packit 98cdb6
Packit 98cdb6
static void gdk_x11_draw_image     (GdkDrawable     *drawable,
Packit 98cdb6
                                    GdkGC           *gc,
Packit 98cdb6
                                    GdkImage        *image,
Packit 98cdb6
                                    gint             xsrc,
Packit 98cdb6
                                    gint             ysrc,
Packit 98cdb6
                                    gint             xdest,
Packit 98cdb6
                                    gint             ydest,
Packit 98cdb6
                                    gint             width,
Packit 98cdb6
                                    gint             height);
Packit 98cdb6
static void gdk_x11_draw_pixbuf    (GdkDrawable     *drawable,
Packit 98cdb6
				    GdkGC           *gc,
Packit 98cdb6
				    GdkPixbuf       *pixbuf,
Packit 98cdb6
				    gint             src_x,
Packit 98cdb6
				    gint             src_y,
Packit 98cdb6
				    gint             dest_x,
Packit 98cdb6
				    gint             dest_y,
Packit 98cdb6
				    gint             width,
Packit 98cdb6
				    gint             height,
Packit 98cdb6
				    GdkRgbDither     dither,
Packit 98cdb6
				    gint             x_dither,
Packit 98cdb6
				    gint             y_dither);
Packit 98cdb6
Packit 98cdb6
static cairo_surface_t *gdk_x11_ref_cairo_surface (GdkDrawable *drawable);
Packit 98cdb6
     
Packit 98cdb6
static void gdk_x11_set_colormap   (GdkDrawable    *drawable,
Packit 98cdb6
                                    GdkColormap    *colormap);
Packit 98cdb6
Packit 98cdb6
static GdkColormap* gdk_x11_get_colormap   (GdkDrawable    *drawable);
Packit 98cdb6
static gint         gdk_x11_get_depth      (GdkDrawable    *drawable);
Packit 98cdb6
static GdkScreen *  gdk_x11_get_screen	   (GdkDrawable    *drawable);
Packit 98cdb6
static GdkVisual*   gdk_x11_get_visual     (GdkDrawable    *drawable);
Packit 98cdb6
Packit 98cdb6
static void gdk_drawable_impl_x11_finalize   (GObject *object);
Packit 98cdb6
Packit 98cdb6
static const cairo_user_data_key_t gdk_x11_cairo_key;
Packit 98cdb6
Packit 98cdb6
G_DEFINE_TYPE (GdkDrawableImplX11, _gdk_drawable_impl_x11, GDK_TYPE_DRAWABLE)
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
_gdk_drawable_impl_x11_class_init (GdkDrawableImplX11Class *klass)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableClass *drawable_class = GDK_DRAWABLE_CLASS (klass);
Packit 98cdb6
  GObjectClass *object_class = G_OBJECT_CLASS (klass);
Packit 98cdb6
  
Packit 98cdb6
  object_class->finalize = gdk_drawable_impl_x11_finalize;
Packit 98cdb6
  
Packit 98cdb6
  drawable_class->create_gc = _gdk_x11_gc_new;
Packit 98cdb6
  drawable_class->draw_rectangle = gdk_x11_draw_rectangle;
Packit 98cdb6
  drawable_class->draw_arc = gdk_x11_draw_arc;
Packit 98cdb6
  drawable_class->draw_polygon = gdk_x11_draw_polygon;
Packit 98cdb6
  drawable_class->draw_text = gdk_x11_draw_text;
Packit 98cdb6
  drawable_class->draw_text_wc = gdk_x11_draw_text_wc;
Packit 98cdb6
  drawable_class->draw_drawable_with_src = gdk_x11_draw_drawable;
Packit 98cdb6
  drawable_class->draw_points = gdk_x11_draw_points;
Packit 98cdb6
  drawable_class->draw_segments = gdk_x11_draw_segments;
Packit 98cdb6
  drawable_class->draw_lines = gdk_x11_draw_lines;
Packit 98cdb6
  drawable_class->draw_image = gdk_x11_draw_image;
Packit 98cdb6
  drawable_class->draw_pixbuf = gdk_x11_draw_pixbuf;
Packit 98cdb6
  
Packit 98cdb6
  drawable_class->ref_cairo_surface = gdk_x11_ref_cairo_surface;
Packit 98cdb6
Packit 98cdb6
  drawable_class->set_colormap = gdk_x11_set_colormap;
Packit 98cdb6
  drawable_class->get_colormap = gdk_x11_get_colormap;
Packit 98cdb6
Packit 98cdb6
  drawable_class->get_depth = gdk_x11_get_depth;
Packit 98cdb6
  drawable_class->get_screen = gdk_x11_get_screen;
Packit 98cdb6
  drawable_class->get_visual = gdk_x11_get_visual;
Packit 98cdb6
  
Packit 98cdb6
  drawable_class->_copy_to_image = _gdk_x11_copy_to_image;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
_gdk_drawable_impl_x11_init (GdkDrawableImplX11 *impl)
Packit 98cdb6
{
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_drawable_impl_x11_finalize (GObject *object)
Packit 98cdb6
{
Packit 98cdb6
  gdk_drawable_set_colormap (GDK_DRAWABLE (object), NULL);
Packit 98cdb6
Packit 98cdb6
  G_OBJECT_CLASS (_gdk_drawable_impl_x11_parent_class)->finalize (object);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * _gdk_x11_drawable_finish:
Packit 98cdb6
 * @drawable: a #GdkDrawableImplX11.
Packit 98cdb6
 * 
Packit 98cdb6
 * Performs necessary cleanup prior to freeing a pixmap or
Packit 98cdb6
 * destroying a window.
Packit 98cdb6
 **/
Packit 98cdb6
void
Packit 98cdb6
_gdk_x11_drawable_finish (GdkDrawable *drawable)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
  
Packit 98cdb6
  if (impl->picture)
Packit 98cdb6
    {
Packit 98cdb6
      XRenderFreePicture (GDK_SCREEN_XDISPLAY (impl->screen),
Packit 98cdb6
			  impl->picture);
Packit 98cdb6
      impl->picture = None;
Packit 98cdb6
    }
Packit 98cdb6
  
Packit 98cdb6
  if (impl->cairo_surface)
Packit 98cdb6
    {
Packit 98cdb6
      cairo_surface_finish (impl->cairo_surface);
Packit 98cdb6
      cairo_surface_set_user_data (impl->cairo_surface, &gdk_x11_cairo_key,
Packit 98cdb6
				   NULL, NULL);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * _gdk_x11_drawable_update_size:
Packit 98cdb6
 * @drawable: a #GdkDrawableImplX11.
Packit 98cdb6
 * 
Packit 98cdb6
 * Updates the state of the drawable (in particular the drawable's
Packit 98cdb6
 * cairo surface) when its size has changed.
Packit 98cdb6
 **/
Packit 98cdb6
void
Packit 98cdb6
_gdk_x11_drawable_update_size (GdkDrawable *drawable)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
  
Packit 98cdb6
  if (impl->cairo_surface)
Packit 98cdb6
    {
Packit 98cdb6
      int width, height;
Packit 98cdb6
      
Packit 98cdb6
      gdk_drawable_get_size (drawable, &width, &height);
Packit 98cdb6
      cairo_xlib_surface_set_size (impl->cairo_surface, width, height);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
try_pixmap (Display *xdisplay,
Packit 98cdb6
	    int      screen,
Packit 98cdb6
	    int      depth)
Packit 98cdb6
{
Packit 98cdb6
  Pixmap pixmap = XCreatePixmap (xdisplay,
Packit 98cdb6
				 RootWindow (xdisplay, screen),
Packit 98cdb6
				 1, 1, depth);
Packit 98cdb6
  XFreePixmap (xdisplay, pixmap);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
gboolean
Packit 98cdb6
_gdk_x11_have_render (GdkDisplay *display)
Packit 98cdb6
{
Packit 98cdb6
  Display *xdisplay = GDK_DISPLAY_XDISPLAY (display);
Packit 98cdb6
  GdkDisplayX11 *x11display = GDK_DISPLAY_X11 (display);
Packit 98cdb6
Packit 98cdb6
  if (x11display->have_render == GDK_UNKNOWN)
Packit 98cdb6
    {
Packit 98cdb6
      int event_base, error_base;
Packit 98cdb6
      x11display->have_render =
Packit 98cdb6
	XRenderQueryExtension (xdisplay, &event_base, &error_base)
Packit 98cdb6
	? GDK_YES : GDK_NO;
Packit 98cdb6
Packit 98cdb6
      if (x11display->have_render == GDK_YES)
Packit 98cdb6
	{
Packit 98cdb6
	  /*
Packit 98cdb6
	   * Sun advertises RENDER, but fails to support 32-bit pixmaps.
Packit 98cdb6
	   * That is just no good.  Therefore, we check all screens
Packit 98cdb6
	   * for proper support.
Packit 98cdb6
	   */
Packit 98cdb6
Packit 98cdb6
	  int screen;
Packit 98cdb6
	  for (screen = 0; screen < ScreenCount (xdisplay); screen++)
Packit 98cdb6
	    {
Packit 98cdb6
	      int count;
Packit 98cdb6
	      int *depths = XListDepths (xdisplay, screen, &count);
Packit 98cdb6
	      gboolean has_8 = FALSE, has_32 = FALSE;
Packit 98cdb6
Packit 98cdb6
	      if (depths)
Packit 98cdb6
		{
Packit 98cdb6
		  int i;
Packit 98cdb6
Packit 98cdb6
		  for (i = 0; i < count; i++)
Packit 98cdb6
		    {
Packit 98cdb6
		      if (depths[i] == 8)
Packit 98cdb6
			has_8 = TRUE;
Packit 98cdb6
		      else if (depths[i] == 32)
Packit 98cdb6
			has_32 = TRUE;
Packit 98cdb6
		    }
Packit 98cdb6
		  XFree (depths);
Packit 98cdb6
		}
Packit 98cdb6
Packit 98cdb6
	      /* At this point, we might have a false positive;
Packit 98cdb6
	       * buggy versions of Xinerama only report depths for
Packit 98cdb6
	       * which there is an associated visual; so we actually
Packit 98cdb6
	       * go ahead and try create pixmaps.
Packit 98cdb6
	       */
Packit 98cdb6
	      if (!(has_8 && has_32))
Packit 98cdb6
		{
Packit 98cdb6
		  gdk_error_trap_push ();
Packit 98cdb6
		  if (!has_8)
Packit 98cdb6
		    try_pixmap (xdisplay, screen, 8);
Packit 98cdb6
		  if (!has_32)
Packit 98cdb6
		    try_pixmap (xdisplay, screen, 32);
Packit 98cdb6
		  XSync (xdisplay, False);
Packit 98cdb6
		  if (gdk_error_trap_pop () == 0)
Packit 98cdb6
		    {
Packit 98cdb6
		      has_8 = TRUE;
Packit 98cdb6
		      has_32 = TRUE;
Packit 98cdb6
		    }
Packit 98cdb6
		}
Packit 98cdb6
	      
Packit 98cdb6
	      if (!(has_8 && has_32))
Packit 98cdb6
		{
Packit 98cdb6
		  g_warning ("The X server advertises that RENDER support is present,\n"
Packit 98cdb6
			     "but fails to supply the necessary pixmap support.  In\n"
Packit 98cdb6
			     "other words, it is buggy.");
Packit 98cdb6
		  x11display->have_render = GDK_NO;
Packit 98cdb6
		  break;
Packit 98cdb6
		}
Packit 98cdb6
	    }
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return x11display->have_render == GDK_YES;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static Picture
Packit 98cdb6
gdk_x11_drawable_get_picture (GdkDrawable *drawable)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
  
Packit 98cdb6
  if (!impl->picture)
Packit 98cdb6
    {
Packit 98cdb6
      Display *xdisplay = GDK_SCREEN_XDISPLAY (impl->screen);
Packit 98cdb6
      XRenderPictFormat *format;
Packit 98cdb6
      
Packit 98cdb6
      GdkVisual *visual = gdk_drawable_get_visual (GDK_DRAWABLE_IMPL_X11 (drawable)->wrapper);
Packit 98cdb6
      if (!visual)
Packit 98cdb6
	return None;
Packit 98cdb6
Packit 98cdb6
      format = XRenderFindVisualFormat (xdisplay, GDK_VISUAL_XVISUAL (visual));
Packit 98cdb6
      if (format)
Packit 98cdb6
	{
Packit 98cdb6
	  XRenderPictureAttributes attributes;
Packit 98cdb6
	  attributes.graphics_exposures = False;
Packit 98cdb6
	  
Packit 98cdb6
	  impl->picture = XRenderCreatePicture (xdisplay, impl->xid, format,
Packit 98cdb6
						CPGraphicsExposure, &attributes);
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
  
Packit 98cdb6
  return impl->picture;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_drawable_update_picture_clip (GdkDrawable *drawable,
Packit 98cdb6
				      GdkGC       *gc)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
  Display *xdisplay = GDK_SCREEN_XDISPLAY (impl->screen);
Packit 98cdb6
  Picture picture = gdk_x11_drawable_get_picture (drawable);
Packit 98cdb6
  GdkRegion *clip_region = gc ? _gdk_gc_get_clip_region (gc) : NULL;
Packit 98cdb6
Packit 98cdb6
  if (clip_region)
Packit 98cdb6
    {
Packit 98cdb6
      GdkRegionBox *boxes = clip_region->rects;
Packit 98cdb6
      gint n_boxes = clip_region->numRects;
Packit 98cdb6
      XRectangle *rects = g_new (XRectangle, n_boxes);
Packit 98cdb6
      int i;
Packit 98cdb6
Packit 98cdb6
      for (i=0; i < n_boxes; i++)
Packit 98cdb6
	{
Packit 98cdb6
	  rects[i].x = CLAMP (boxes[i].x1 + gc->clip_x_origin, G_MINSHORT, G_MAXSHORT);
Packit 98cdb6
	  rects[i].y = CLAMP (boxes[i].y1 + gc->clip_y_origin, G_MINSHORT, G_MAXSHORT);
Packit 98cdb6
	  rects[i].width = CLAMP (boxes[i].x2 + gc->clip_x_origin, G_MINSHORT, G_MAXSHORT) - rects[i].x;
Packit 98cdb6
	  rects[i].height = CLAMP (boxes[i].y2 + gc->clip_y_origin, G_MINSHORT, G_MAXSHORT) - rects[i].y;
Packit 98cdb6
	}
Packit 98cdb6
      
Packit 98cdb6
      XRenderSetPictureClipRectangles (xdisplay, picture,
Packit 98cdb6
				       0, 0, rects, n_boxes);
Packit 98cdb6
      
Packit 98cdb6
      g_free (rects);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      XRenderPictureAttributes pa;
Packit 98cdb6
      GdkBitmap *mask;
Packit 98cdb6
      gulong pa_mask;
Packit 98cdb6
Packit 98cdb6
      pa_mask = CPClipMask;
Packit 98cdb6
      if (gc && (mask = _gdk_gc_get_clip_mask (gc)))
Packit 98cdb6
	{
Packit 98cdb6
	  pa.clip_mask = GDK_PIXMAP_XID (mask);
Packit 98cdb6
	  pa.clip_x_origin = gc->clip_x_origin;
Packit 98cdb6
	  pa.clip_y_origin = gc->clip_y_origin;
Packit 98cdb6
	  pa_mask |= CPClipXOrigin | CPClipYOrigin;
Packit 98cdb6
	}
Packit 98cdb6
      else
Packit 98cdb6
	pa.clip_mask = None;
Packit 98cdb6
Packit 98cdb6
      XRenderChangePicture (xdisplay, picture,
Packit 98cdb6
			    pa_mask, &pa);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/*****************************************************
Packit 98cdb6
 * X11 specific implementations of generic functions *
Packit 98cdb6
 *****************************************************/
Packit 98cdb6
Packit 98cdb6
static GdkColormap*
Packit 98cdb6
gdk_x11_get_colormap (GdkDrawable *drawable)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
Packit 98cdb6
  return impl->colormap;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_set_colormap (GdkDrawable *drawable,
Packit 98cdb6
                      GdkColormap *colormap)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
Packit 98cdb6
  if (impl->colormap == colormap)
Packit 98cdb6
    return;
Packit 98cdb6
  
Packit 98cdb6
  if (impl->colormap)
Packit 98cdb6
    g_object_unref (impl->colormap);
Packit 98cdb6
  impl->colormap = colormap;
Packit 98cdb6
  if (impl->colormap)
Packit 98cdb6
    g_object_ref (impl->colormap);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/* Drawing
Packit 98cdb6
 */
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_draw_rectangle (GdkDrawable *drawable,
Packit 98cdb6
			GdkGC       *gc,
Packit 98cdb6
			gboolean     filled,
Packit 98cdb6
			gint         x,
Packit 98cdb6
			gint         y,
Packit 98cdb6
			gint         width,
Packit 98cdb6
			gint         height)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
  
Packit 98cdb6
  if (filled)
Packit 98cdb6
    XFillRectangle (GDK_SCREEN_XDISPLAY (impl->screen), impl->xid,
Packit 98cdb6
		    GDK_GC_GET_XGC (gc), x, y, width, height);
Packit 98cdb6
  else
Packit 98cdb6
    XDrawRectangle (GDK_SCREEN_XDISPLAY (impl->screen), impl->xid,
Packit 98cdb6
		    GDK_GC_GET_XGC (gc), x, y, width, height);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_draw_arc (GdkDrawable *drawable,
Packit 98cdb6
		  GdkGC       *gc,
Packit 98cdb6
		  gboolean     filled,
Packit 98cdb6
		  gint         x,
Packit 98cdb6
		  gint         y,
Packit 98cdb6
		  gint         width,
Packit 98cdb6
		  gint         height,
Packit 98cdb6
		  gint         angle1,
Packit 98cdb6
		  gint         angle2)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
Packit 98cdb6
  
Packit 98cdb6
  if (filled)
Packit 98cdb6
    XFillArc (GDK_SCREEN_XDISPLAY (impl->screen), impl->xid,
Packit 98cdb6
	      GDK_GC_GET_XGC (gc), x, y, width, height, angle1, angle2);
Packit 98cdb6
  else
Packit 98cdb6
    XDrawArc (GDK_SCREEN_XDISPLAY (impl->screen), impl->xid,
Packit 98cdb6
	      GDK_GC_GET_XGC (gc), x, y, width, height, angle1, angle2);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_draw_polygon (GdkDrawable *drawable,
Packit 98cdb6
		      GdkGC       *gc,
Packit 98cdb6
		      gboolean     filled,
Packit 98cdb6
		      GdkPoint    *points,
Packit 98cdb6
		      gint         npoints)
Packit 98cdb6
{
Packit 98cdb6
  XPoint *tmp_points;
Packit 98cdb6
  gint tmp_npoints, i;
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
Packit 98cdb6
  
Packit 98cdb6
  if (!filled &&
Packit 98cdb6
      (points[0].x != points[npoints-1].x || points[0].y != points[npoints-1].y))
Packit 98cdb6
    {
Packit 98cdb6
      tmp_npoints = npoints + 1;
Packit 98cdb6
      tmp_points = g_new (XPoint, tmp_npoints);
Packit 98cdb6
      tmp_points[npoints].x = points[0].x;
Packit 98cdb6
      tmp_points[npoints].y = points[0].y;
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      tmp_npoints = npoints;
Packit 98cdb6
      tmp_points = g_new (XPoint, tmp_npoints);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  for (i=0; i
Packit 98cdb6
    {
Packit 98cdb6
      tmp_points[i].x = points[i].x;
Packit 98cdb6
      tmp_points[i].y = points[i].y;
Packit 98cdb6
    }
Packit 98cdb6
  
Packit 98cdb6
  if (filled)
Packit 98cdb6
    XFillPolygon (GDK_SCREEN_XDISPLAY (impl->screen), impl->xid,
Packit 98cdb6
		  GDK_GC_GET_XGC (gc), tmp_points, tmp_npoints, Complex, CoordModeOrigin);
Packit 98cdb6
  else
Packit 98cdb6
    XDrawLines (GDK_SCREEN_XDISPLAY (impl->screen), impl->xid,
Packit 98cdb6
		GDK_GC_GET_XGC (gc), tmp_points, tmp_npoints, CoordModeOrigin);
Packit 98cdb6
Packit 98cdb6
  g_free (tmp_points);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/* gdk_x11_draw_text
Packit 98cdb6
 *
Packit 98cdb6
 * Modified by Li-Da Lho to draw 16 bits and Multibyte strings
Packit 98cdb6
 *
Packit 98cdb6
 * Interface changed: add "GdkFont *font" to specify font or fontset explicitely
Packit 98cdb6
 */
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_draw_text (GdkDrawable *drawable,
Packit 98cdb6
		   GdkFont     *font,
Packit 98cdb6
		   GdkGC       *gc,
Packit 98cdb6
		   gint         x,
Packit 98cdb6
		   gint         y,
Packit 98cdb6
		   const gchar *text,
Packit 98cdb6
		   gint         text_length)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
  Display *xdisplay;
Packit 98cdb6
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
  xdisplay = GDK_SCREEN_XDISPLAY (impl->screen);
Packit 98cdb6
  
Packit 98cdb6
  if (font->type == GDK_FONT_FONT)
Packit 98cdb6
    {
Packit 98cdb6
      XFontStruct *xfont = (XFontStruct *) GDK_FONT_XFONT (font);
Packit 98cdb6
      XSetFont(xdisplay, GDK_GC_GET_XGC (gc), xfont->fid);
Packit 98cdb6
      if ((xfont->min_byte1 == 0) && (xfont->max_byte1 == 0))
Packit 98cdb6
	{
Packit 98cdb6
	  XDrawString (xdisplay, impl->xid,
Packit 98cdb6
		       GDK_GC_GET_XGC (gc), x, y, text, text_length);
Packit 98cdb6
	}
Packit 98cdb6
      else
Packit 98cdb6
	{
Packit 98cdb6
	  XDrawString16 (xdisplay, impl->xid,
Packit 98cdb6
			 GDK_GC_GET_XGC (gc), x, y, (XChar2b *) text, text_length / 2);
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
  else if (font->type == GDK_FONT_FONTSET)
Packit 98cdb6
    {
Packit 98cdb6
      XFontSet fontset = (XFontSet) GDK_FONT_XFONT (font);
Packit 98cdb6
      XmbDrawString (xdisplay, impl->xid,
Packit 98cdb6
		     fontset, GDK_GC_GET_XGC (gc), x, y, text, text_length);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    g_error("undefined font type\n");
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_draw_text_wc (GdkDrawable    *drawable,
Packit 98cdb6
		      GdkFont	     *font,
Packit 98cdb6
		      GdkGC	     *gc,
Packit 98cdb6
		      gint	      x,
Packit 98cdb6
		      gint	      y,
Packit 98cdb6
		      const GdkWChar *text,
Packit 98cdb6
		      gint	      text_length)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
  Display *xdisplay;
Packit 98cdb6
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
  xdisplay = GDK_SCREEN_XDISPLAY (impl->screen);
Packit 98cdb6
  
Packit 98cdb6
  if (font->type == GDK_FONT_FONT)
Packit 98cdb6
    {
Packit 98cdb6
      XFontStruct *xfont = (XFontStruct *) GDK_FONT_XFONT (font);
Packit 98cdb6
      gchar *text_8bit;
Packit 98cdb6
      gint i;
Packit 98cdb6
      XSetFont(xdisplay, GDK_GC_GET_XGC (gc), xfont->fid);
Packit 98cdb6
      text_8bit = g_new (gchar, text_length);
Packit 98cdb6
      for (i=0; i
Packit 98cdb6
      XDrawString (xdisplay, impl->xid,
Packit 98cdb6
                   GDK_GC_GET_XGC (gc), x, y, text_8bit, text_length);
Packit 98cdb6
      g_free (text_8bit);
Packit 98cdb6
    }
Packit 98cdb6
  else if (font->type == GDK_FONT_FONTSET)
Packit 98cdb6
    {
Packit 98cdb6
      if (sizeof(GdkWChar) == sizeof(wchar_t))
Packit 98cdb6
	{
Packit 98cdb6
	  XwcDrawString (xdisplay, impl->xid,
Packit 98cdb6
			 (XFontSet) GDK_FONT_XFONT (font),
Packit 98cdb6
			 GDK_GC_GET_XGC (gc), x, y, (wchar_t *)text, text_length);
Packit 98cdb6
	}
Packit 98cdb6
      else
Packit 98cdb6
	{
Packit 98cdb6
	  wchar_t *text_wchar;
Packit 98cdb6
	  gint i;
Packit 98cdb6
	  text_wchar = g_new (wchar_t, text_length);
Packit 98cdb6
	  for (i=0; i
Packit 98cdb6
	  XwcDrawString (xdisplay, impl->xid,
Packit 98cdb6
			 (XFontSet) GDK_FONT_XFONT (font),
Packit 98cdb6
			 GDK_GC_GET_XGC (gc), x, y, text_wchar, text_length);
Packit 98cdb6
	  g_free (text_wchar);
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    g_error("undefined font type\n");
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_draw_drawable (GdkDrawable *drawable,
Packit 98cdb6
		       GdkGC       *gc,
Packit 98cdb6
		       GdkPixmap   *src,
Packit 98cdb6
		       gint         xsrc,
Packit 98cdb6
		       gint         ysrc,
Packit 98cdb6
		       gint         xdest,
Packit 98cdb6
		       gint         ydest,
Packit 98cdb6
		       gint         width,
Packit 98cdb6
		       gint         height,
Packit 98cdb6
		       GdkDrawable *original_src)
Packit 98cdb6
{
Packit 98cdb6
  int src_depth = gdk_drawable_get_depth (src);
Packit 98cdb6
  int dest_depth = gdk_drawable_get_depth (drawable);
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
  GdkDrawableImplX11 *src_impl;
Packit 98cdb6
  
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
Packit 98cdb6
  if (GDK_IS_DRAWABLE_IMPL_X11 (src))
Packit 98cdb6
    src_impl = GDK_DRAWABLE_IMPL_X11 (src);
Packit 98cdb6
  else if (GDK_IS_WINDOW (src))
Packit 98cdb6
    src_impl = GDK_DRAWABLE_IMPL_X11(((GdkWindowObject *)src)->impl);
Packit 98cdb6
  else
Packit 98cdb6
    src_impl = GDK_DRAWABLE_IMPL_X11(((GdkPixmapObject *)src)->impl);
Packit 98cdb6
Packit 98cdb6
  if (GDK_IS_WINDOW_IMPL_X11 (impl) &&
Packit 98cdb6
      GDK_IS_PIXMAP_IMPL_X11 (src_impl))
Packit 98cdb6
    {
Packit 98cdb6
      GdkPixmapImplX11 *src_pixmap = GDK_PIXMAP_IMPL_X11 (src_impl);
Packit 98cdb6
      /* Work around an Xserver bug where non-visible areas from
Packit 98cdb6
       * a pixmap to a window will clear the window background
Packit 98cdb6
       * in destination areas that are supposed to be clipped out.
Packit 98cdb6
       * This is a problem with client side windows as this means
Packit 98cdb6
       * things may draw outside the virtual windows. This could
Packit 98cdb6
       * also happen for window to window copies, but I don't
Packit 98cdb6
       * think we generate any calls like that.
Packit 98cdb6
       *
Packit 98cdb6
       * See: 
Packit 98cdb6
       * http://lists.freedesktop.org/archives/xorg/2009-February/043318.html
Packit 98cdb6
       */
Packit 98cdb6
      if (xsrc < 0)
Packit 98cdb6
	{
Packit 98cdb6
	  width += xsrc;
Packit 98cdb6
	  xdest -= xsrc;
Packit 98cdb6
	  xsrc = 0;
Packit 98cdb6
	}
Packit 98cdb6
      
Packit 98cdb6
      if (ysrc < 0)
Packit 98cdb6
	{
Packit 98cdb6
	  height += ysrc;
Packit 98cdb6
	  ydest -= ysrc;
Packit 98cdb6
	  ysrc = 0;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      if (xsrc + width > src_pixmap->width)
Packit 98cdb6
	width = src_pixmap->width - xsrc;
Packit 98cdb6
      if (ysrc + height > src_pixmap->height)
Packit 98cdb6
	height = src_pixmap->height - ysrc;
Packit 98cdb6
    }
Packit 98cdb6
  
Packit 98cdb6
  if (src_depth == 1)
Packit 98cdb6
    {
Packit 98cdb6
      XCopyArea (GDK_SCREEN_XDISPLAY (impl->screen),
Packit 98cdb6
                 src_impl->xid,
Packit 98cdb6
		 impl->xid,
Packit 98cdb6
		 GDK_GC_GET_XGC (gc),
Packit 98cdb6
		 xsrc, ysrc,
Packit 98cdb6
		 width, height,
Packit 98cdb6
		 xdest, ydest);
Packit 98cdb6
    }
Packit 98cdb6
  else if (dest_depth != 0 && src_depth == dest_depth)
Packit 98cdb6
    {
Packit 98cdb6
      XCopyArea (GDK_SCREEN_XDISPLAY (impl->screen),
Packit 98cdb6
                 src_impl->xid,
Packit 98cdb6
		 impl->xid,
Packit 98cdb6
		 GDK_GC_GET_XGC (gc),
Packit 98cdb6
		 xsrc, ysrc,
Packit 98cdb6
		 width, height,
Packit 98cdb6
		 xdest, ydest);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    g_warning ("Attempt to draw a drawable with depth %d to a drawable with depth %d",
Packit 98cdb6
               src_depth, dest_depth);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_draw_points (GdkDrawable *drawable,
Packit 98cdb6
		     GdkGC       *gc,
Packit 98cdb6
		     GdkPoint    *points,
Packit 98cdb6
		     gint         npoints)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
Packit 98cdb6
  
Packit 98cdb6
  /* We special-case npoints == 1, because X will merge multiple
Packit 98cdb6
   * consecutive XDrawPoint requests into a PolyPoint request
Packit 98cdb6
   */
Packit 98cdb6
  if (npoints == 1)
Packit 98cdb6
    {
Packit 98cdb6
      XDrawPoint (GDK_SCREEN_XDISPLAY (impl->screen),
Packit 98cdb6
		  impl->xid,
Packit 98cdb6
		  GDK_GC_GET_XGC (gc),
Packit 98cdb6
		  points[0].x, points[0].y);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      gint i;
Packit 98cdb6
      XPoint *tmp_points = g_new (XPoint, npoints);
Packit 98cdb6
Packit 98cdb6
      for (i=0; i
Packit 98cdb6
	{
Packit 98cdb6
	  tmp_points[i].x = points[i].x;
Packit 98cdb6
	  tmp_points[i].y = points[i].y;
Packit 98cdb6
	}
Packit 98cdb6
      
Packit 98cdb6
      XDrawPoints (GDK_SCREEN_XDISPLAY (impl->screen),
Packit 98cdb6
		   impl->xid,
Packit 98cdb6
		   GDK_GC_GET_XGC (gc),
Packit 98cdb6
		   tmp_points,
Packit 98cdb6
		   npoints,
Packit 98cdb6
		   CoordModeOrigin);
Packit 98cdb6
Packit 98cdb6
      g_free (tmp_points);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_draw_segments (GdkDrawable *drawable,
Packit 98cdb6
		       GdkGC       *gc,
Packit 98cdb6
		       GdkSegment  *segs,
Packit 98cdb6
		       gint         nsegs)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
Packit 98cdb6
  
Packit 98cdb6
  /* We special-case nsegs == 1, because X will merge multiple
Packit 98cdb6
   * consecutive XDrawLine requests into a PolySegment request
Packit 98cdb6
   */
Packit 98cdb6
  if (nsegs == 1)
Packit 98cdb6
    {
Packit 98cdb6
      XDrawLine (GDK_SCREEN_XDISPLAY (impl->screen), impl->xid,
Packit 98cdb6
		 GDK_GC_GET_XGC (gc), segs[0].x1, segs[0].y1,
Packit 98cdb6
		 segs[0].x2, segs[0].y2);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      gint i;
Packit 98cdb6
      XSegment *tmp_segs = g_new (XSegment, nsegs);
Packit 98cdb6
Packit 98cdb6
      for (i=0; i
Packit 98cdb6
	{
Packit 98cdb6
	  tmp_segs[i].x1 = segs[i].x1;
Packit 98cdb6
	  tmp_segs[i].x2 = segs[i].x2;
Packit 98cdb6
	  tmp_segs[i].y1 = segs[i].y1;
Packit 98cdb6
	  tmp_segs[i].y2 = segs[i].y2;
Packit 98cdb6
	}
Packit 98cdb6
      
Packit 98cdb6
      XDrawSegments (GDK_SCREEN_XDISPLAY (impl->screen),
Packit 98cdb6
		     impl->xid,
Packit 98cdb6
		     GDK_GC_GET_XGC (gc),
Packit 98cdb6
		     tmp_segs, nsegs);
Packit 98cdb6
Packit 98cdb6
      g_free (tmp_segs);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_draw_lines (GdkDrawable *drawable,
Packit 98cdb6
		    GdkGC       *gc,
Packit 98cdb6
		    GdkPoint    *points,
Packit 98cdb6
		    gint         npoints)
Packit 98cdb6
{
Packit 98cdb6
  gint i;
Packit 98cdb6
  XPoint *tmp_points = g_new (XPoint, npoints);
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
Packit 98cdb6
  
Packit 98cdb6
  for (i=0; i
Packit 98cdb6
    {
Packit 98cdb6
      tmp_points[i].x = points[i].x;
Packit 98cdb6
      tmp_points[i].y = points[i].y;
Packit 98cdb6
    }
Packit 98cdb6
      
Packit 98cdb6
  XDrawLines (GDK_SCREEN_XDISPLAY (impl->screen),
Packit 98cdb6
	      impl->xid,
Packit 98cdb6
	      GDK_GC_GET_XGC (gc),
Packit 98cdb6
	      tmp_points, npoints,
Packit 98cdb6
	      CoordModeOrigin);
Packit 98cdb6
Packit 98cdb6
  g_free (tmp_points);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_draw_image     (GdkDrawable     *drawable,
Packit 98cdb6
                        GdkGC           *gc,
Packit 98cdb6
                        GdkImage        *image,
Packit 98cdb6
                        gint             xsrc,
Packit 98cdb6
                        gint             ysrc,
Packit 98cdb6
                        gint             xdest,
Packit 98cdb6
                        gint             ydest,
Packit 98cdb6
                        gint             width,
Packit 98cdb6
                        gint             height)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
Packit 98cdb6
#ifdef USE_SHM  
Packit 98cdb6
  if (image->type == GDK_IMAGE_SHARED)
Packit 98cdb6
    XShmPutImage (GDK_SCREEN_XDISPLAY (impl->screen), impl->xid,
Packit 98cdb6
                  GDK_GC_GET_XGC (gc), GDK_IMAGE_XIMAGE (image),
Packit 98cdb6
                  xsrc, ysrc, xdest, ydest, width, height, False);
Packit 98cdb6
  else
Packit 98cdb6
#endif
Packit 98cdb6
    XPutImage (GDK_SCREEN_XDISPLAY (impl->screen), impl->xid,
Packit 98cdb6
               GDK_GC_GET_XGC (gc), GDK_IMAGE_XIMAGE (image),
Packit 98cdb6
               xsrc, ysrc, xdest, ydest, width, height);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gint
Packit 98cdb6
gdk_x11_get_depth (GdkDrawable *drawable)
Packit 98cdb6
{
Packit 98cdb6
  /* This is a bit bogus but I'm not sure the other way is better */
Packit 98cdb6
Packit 98cdb6
  return gdk_drawable_get_depth (GDK_DRAWABLE_IMPL_X11 (drawable)->wrapper);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static GdkDrawable *
Packit 98cdb6
get_impl_drawable (GdkDrawable *drawable)
Packit 98cdb6
{
Packit 98cdb6
  if (GDK_IS_WINDOW (drawable))
Packit 98cdb6
    return ((GdkWindowObject *)drawable)->impl;
Packit 98cdb6
  else if (GDK_IS_PIXMAP (drawable))
Packit 98cdb6
    return ((GdkPixmapObject *)drawable)->impl;
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      g_warning (G_STRLOC " drawable is not a pixmap or window");
Packit 98cdb6
      return NULL;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static GdkScreen*
Packit 98cdb6
gdk_x11_get_screen (GdkDrawable *drawable)
Packit 98cdb6
{
Packit 98cdb6
  if (GDK_IS_DRAWABLE_IMPL_X11 (drawable))
Packit 98cdb6
    return GDK_DRAWABLE_IMPL_X11 (drawable)->screen;
Packit 98cdb6
  else
Packit 98cdb6
    return GDK_DRAWABLE_IMPL_X11 (get_impl_drawable (drawable))->screen;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static GdkVisual*
Packit 98cdb6
gdk_x11_get_visual (GdkDrawable    *drawable)
Packit 98cdb6
{
Packit 98cdb6
  return gdk_drawable_get_visual (GDK_DRAWABLE_IMPL_X11 (drawable)->wrapper);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gdk_x11_drawable_get_xdisplay:
Packit 98cdb6
 * @drawable: a #GdkDrawable.
Packit 98cdb6
 * 
Packit 98cdb6
 * Returns the display of a #GdkDrawable.
Packit 98cdb6
 * 
Packit 98cdb6
 * Return value: an Xlib <type>Display*</type>.
Packit 98cdb6
 **/
Packit 98cdb6
Display *
Packit 98cdb6
gdk_x11_drawable_get_xdisplay (GdkDrawable *drawable)
Packit 98cdb6
{
Packit 98cdb6
  if (GDK_IS_DRAWABLE_IMPL_X11 (drawable))
Packit 98cdb6
    return GDK_SCREEN_XDISPLAY (GDK_DRAWABLE_IMPL_X11 (drawable)->screen);
Packit 98cdb6
  else
Packit 98cdb6
    return GDK_SCREEN_XDISPLAY (GDK_DRAWABLE_IMPL_X11 (get_impl_drawable (drawable))->screen);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gdk_x11_drawable_get_xid:
Packit 98cdb6
 * @drawable: a #GdkDrawable.
Packit 98cdb6
 * 
Packit 98cdb6
 * Returns the X resource (window or pixmap) belonging to a #GdkDrawable.
Packit 98cdb6
 * 
Packit 98cdb6
 * Return value: the ID of @drawable's X resource.
Packit 98cdb6
 **/
Packit 98cdb6
XID
Packit 98cdb6
gdk_x11_drawable_get_xid (GdkDrawable *drawable)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawable *impl;
Packit 98cdb6
  
Packit 98cdb6
  if (GDK_IS_WINDOW (drawable))
Packit 98cdb6
    {
Packit 98cdb6
      GdkWindow *window = (GdkWindow *)drawable;
Packit 98cdb6
      
Packit 98cdb6
      /* Try to ensure the window has a native window */
Packit 98cdb6
      if (!_gdk_window_has_impl (window))
Packit 98cdb6
	{
Packit 98cdb6
	  gdk_window_ensure_native (window);
Packit 98cdb6
Packit 98cdb6
	  /* We sync here to ensure the window is created in the Xserver when
Packit 98cdb6
	   * this function returns. This is required because the returned XID
Packit 98cdb6
	   * for this window must be valid immediately, even with another
Packit 98cdb6
	   * connection to the Xserver */
Packit 98cdb6
	  gdk_display_sync (gdk_drawable_get_display (window));
Packit 98cdb6
	}
Packit 98cdb6
      
Packit 98cdb6
      if (!GDK_WINDOW_IS_X11 (window))
Packit 98cdb6
        {
Packit 98cdb6
          g_warning (G_STRLOC " drawable is not a native X11 window");
Packit 98cdb6
          return None;
Packit 98cdb6
        }
Packit 98cdb6
      
Packit 98cdb6
      impl = ((GdkWindowObject *)drawable)->impl;
Packit 98cdb6
    }
Packit 98cdb6
  else if (GDK_IS_PIXMAP (drawable))
Packit 98cdb6
    impl = ((GdkPixmapObject *)drawable)->impl;
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      g_warning (G_STRLOC " drawable is not a pixmap or window");
Packit 98cdb6
      return None;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return ((GdkDrawableImplX11 *)impl)->xid;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
GdkDrawable *
Packit 98cdb6
gdk_x11_window_get_drawable_impl (GdkWindow *window)
Packit 98cdb6
{
Packit 98cdb6
  return ((GdkWindowObject *)window)->impl;
Packit 98cdb6
}
Packit 98cdb6
GdkDrawable *
Packit 98cdb6
gdk_x11_pixmap_get_drawable_impl (GdkPixmap *pixmap)
Packit 98cdb6
{
Packit 98cdb6
  return ((GdkPixmapObject *)pixmap)->impl;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/* Code for accelerated alpha compositing using the RENDER extension.
Packit 98cdb6
 * It's a bit long because there are lots of possibilities for
Packit 98cdb6
 * what's the fastest depending on the available picture formats,
Packit 98cdb6
 * whether we can used shared pixmaps, etc.
Packit 98cdb6
 */
Packit 98cdb6
Packit 98cdb6
static GdkX11FormatType
Packit 98cdb6
select_format (GdkDisplay         *display,
Packit 98cdb6
	       XRenderPictFormat **format,
Packit 98cdb6
	       XRenderPictFormat **mask)
Packit 98cdb6
{
Packit 98cdb6
  Display *xdisplay = GDK_DISPLAY_XDISPLAY (display);
Packit 98cdb6
  XRenderPictFormat pf;
Packit 98cdb6
Packit 98cdb6
  if (!_gdk_x11_have_render (display))
Packit 98cdb6
    return GDK_X11_FORMAT_NONE;
Packit 98cdb6
  
Packit 98cdb6
  /* Look for a 32-bit xRGB and Axxx formats that exactly match the
Packit 98cdb6
   * in memory data format. We can use them as pixmap and mask
Packit 98cdb6
   * to deal with non-premultiplied data.
Packit 98cdb6
   */
Packit 98cdb6
Packit 98cdb6
  pf.type = PictTypeDirect;
Packit 98cdb6
  pf.depth = 32;
Packit 98cdb6
  pf.direct.redMask = 0xff;
Packit 98cdb6
  pf.direct.greenMask = 0xff;
Packit 98cdb6
  pf.direct.blueMask = 0xff;
Packit 98cdb6
  
Packit 98cdb6
  pf.direct.alphaMask = 0;
Packit 98cdb6
  if (ImageByteOrder (xdisplay) == LSBFirst)
Packit 98cdb6
    {
Packit 98cdb6
      /* ABGR */
Packit 98cdb6
      pf.direct.red = 0;
Packit 98cdb6
      pf.direct.green = 8;
Packit 98cdb6
      pf.direct.blue = 16;
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      /* RGBA */
Packit 98cdb6
      pf.direct.red = 24;
Packit 98cdb6
      pf.direct.green = 16;
Packit 98cdb6
      pf.direct.blue = 8;
Packit 98cdb6
    }
Packit 98cdb6
  
Packit 98cdb6
  *format = XRenderFindFormat (xdisplay,
Packit 98cdb6
			       (PictFormatType | PictFormatDepth |
Packit 98cdb6
				PictFormatRedMask | PictFormatRed |
Packit 98cdb6
				PictFormatGreenMask | PictFormatGreen |
Packit 98cdb6
				PictFormatBlueMask | PictFormatBlue |
Packit 98cdb6
				PictFormatAlphaMask),
Packit 98cdb6
			       &pf,
Packit 98cdb6
			       0);
Packit 98cdb6
Packit 98cdb6
  pf.direct.alphaMask = 0xff;
Packit 98cdb6
  if (ImageByteOrder (xdisplay) == LSBFirst)
Packit 98cdb6
    {
Packit 98cdb6
      /* ABGR */
Packit 98cdb6
      pf.direct.alpha = 24;
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      pf.direct.alpha = 0;
Packit 98cdb6
    }
Packit 98cdb6
  
Packit 98cdb6
  *mask = XRenderFindFormat (xdisplay,
Packit 98cdb6
			     (PictFormatType | PictFormatDepth |
Packit 98cdb6
			      PictFormatAlphaMask | PictFormatAlpha),
Packit 98cdb6
			     &pf,
Packit 98cdb6
			     0);
Packit 98cdb6
Packit 98cdb6
  if (*format && *mask)
Packit 98cdb6
    return GDK_X11_FORMAT_EXACT_MASK;
Packit 98cdb6
Packit 98cdb6
  /* OK, that failed, now look for xRGB and Axxx formats in
Packit 98cdb6
   * RENDER's preferred order
Packit 98cdb6
   */
Packit 98cdb6
  pf.direct.alphaMask = 0;
Packit 98cdb6
  /* ARGB */
Packit 98cdb6
  pf.direct.red = 16;
Packit 98cdb6
  pf.direct.green = 8;
Packit 98cdb6
  pf.direct.blue = 0;
Packit 98cdb6
  
Packit 98cdb6
  *format = XRenderFindFormat (xdisplay,
Packit 98cdb6
			       (PictFormatType | PictFormatDepth |
Packit 98cdb6
				PictFormatRedMask | PictFormatRed |
Packit 98cdb6
				PictFormatGreenMask | PictFormatGreen |
Packit 98cdb6
				PictFormatBlueMask | PictFormatBlue |
Packit 98cdb6
				PictFormatAlphaMask),
Packit 98cdb6
			       &pf,
Packit 98cdb6
			       0);
Packit 98cdb6
Packit 98cdb6
  pf.direct.alphaMask = 0xff;
Packit 98cdb6
  pf.direct.alpha = 24;
Packit 98cdb6
  
Packit 98cdb6
  *mask = XRenderFindFormat (xdisplay,
Packit 98cdb6
			     (PictFormatType | PictFormatDepth |
Packit 98cdb6
			      PictFormatAlphaMask | PictFormatAlpha),
Packit 98cdb6
			     &pf,
Packit 98cdb6
			     0);
Packit 98cdb6
Packit 98cdb6
  if (*format && *mask)
Packit 98cdb6
    return GDK_X11_FORMAT_ARGB_MASK;
Packit 98cdb6
Packit 98cdb6
  /* Finally, if neither of the above worked, fall back to
Packit 98cdb6
   * looking for combined ARGB -- we'll premultiply ourselves.
Packit 98cdb6
   */
Packit 98cdb6
Packit 98cdb6
  pf.type = PictTypeDirect;
Packit 98cdb6
  pf.depth = 32;
Packit 98cdb6
  pf.direct.red = 16;
Packit 98cdb6
  pf.direct.green = 8;
Packit 98cdb6
  pf.direct.blue = 0;
Packit 98cdb6
  pf.direct.alphaMask = 0xff;
Packit 98cdb6
  pf.direct.alpha = 24;
Packit 98cdb6
Packit 98cdb6
  *format = XRenderFindFormat (xdisplay,
Packit 98cdb6
			       (PictFormatType | PictFormatDepth |
Packit 98cdb6
				PictFormatRedMask | PictFormatRed |
Packit 98cdb6
				PictFormatGreenMask | PictFormatGreen |
Packit 98cdb6
				PictFormatBlueMask | PictFormatBlue |
Packit 98cdb6
				PictFormatAlphaMask | PictFormatAlpha),
Packit 98cdb6
			       &pf,
Packit 98cdb6
			       0);
Packit 98cdb6
  *mask = NULL;
Packit 98cdb6
Packit 98cdb6
  if (*format)
Packit 98cdb6
    return GDK_X11_FORMAT_ARGB;
Packit 98cdb6
Packit 98cdb6
  return GDK_X11_FORMAT_NONE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
#if 0
Packit 98cdb6
static void
Packit 98cdb6
list_formats (XRenderPictFormat *pf)
Packit 98cdb6
{
Packit 98cdb6
  gint i;
Packit 98cdb6
  
Packit 98cdb6
  for (i=0 ;; i++)
Packit 98cdb6
    {
Packit 98cdb6
      XRenderPictFormat *pf = XRenderFindFormat (impl->xdisplay, 0, NULL, i);
Packit 98cdb6
      if (pf)
Packit 98cdb6
	{
Packit 98cdb6
	  g_print ("%2d R-%#06x/%#06x G-%#06x/%#06x B-%#06x/%#06x A-%#06x/%#06x\n",
Packit 98cdb6
		   pf->depth,
Packit 98cdb6
		   pf->direct.red,
Packit 98cdb6
		   pf->direct.redMask,
Packit 98cdb6
		   pf->direct.green,
Packit 98cdb6
		   pf->direct.greenMask,
Packit 98cdb6
		   pf->direct.blue,
Packit 98cdb6
		   pf->direct.blueMask,
Packit 98cdb6
		   pf->direct.alpha,
Packit 98cdb6
		   pf->direct.alphaMask);
Packit 98cdb6
	}
Packit 98cdb6
      else
Packit 98cdb6
	break;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
#endif  
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
_gdk_x11_convert_to_format (guchar           *src_buf,
Packit 98cdb6
                            gint              src_rowstride,
Packit 98cdb6
                            guchar           *dest_buf,
Packit 98cdb6
                            gint              dest_rowstride,
Packit 98cdb6
                            GdkX11FormatType  dest_format,
Packit 98cdb6
                            GdkByteOrder      dest_byteorder,
Packit 98cdb6
                            gint              width,
Packit 98cdb6
                            gint              height)
Packit 98cdb6
{
Packit 98cdb6
  gint i;
Packit 98cdb6
Packit 98cdb6
  for (i=0; i < height; i++)
Packit 98cdb6
    {
Packit 98cdb6
      switch (dest_format)
Packit 98cdb6
	{
Packit 98cdb6
	case GDK_X11_FORMAT_EXACT_MASK:
Packit 98cdb6
	  {
Packit 98cdb6
	    memcpy (dest_buf + i * dest_rowstride,
Packit 98cdb6
		    src_buf + i * src_rowstride,
Packit 98cdb6
		    width * 4);
Packit 98cdb6
	    break;
Packit 98cdb6
	  }
Packit 98cdb6
	case GDK_X11_FORMAT_ARGB_MASK:
Packit 98cdb6
	  {
Packit 98cdb6
	    guchar *row = src_buf + i * src_rowstride;
Packit 98cdb6
	    if (((gsize)row & 3) != 0)
Packit 98cdb6
	      {
Packit 98cdb6
		guchar *p = row;
Packit 98cdb6
		guint32 *q = (guint32 *)(dest_buf + i * dest_rowstride);
Packit 98cdb6
		guchar *end = p + 4 * width;
Packit 98cdb6
Packit 98cdb6
		while (p < end)
Packit 98cdb6
		  {
Packit 98cdb6
		    *q = (p[3] << 24) | (p[0] << 16) | (p[1] << 8) | p[2];
Packit 98cdb6
		    p += 4;
Packit 98cdb6
		    q++;
Packit 98cdb6
		  }
Packit 98cdb6
	      }
Packit 98cdb6
	    else
Packit 98cdb6
	      {
Packit 98cdb6
		guint32 *p = (guint32 *)row;
Packit 98cdb6
		guint32 *q = (guint32 *)(dest_buf + i * dest_rowstride);
Packit 98cdb6
		guint32 *end = p + width;
Packit 98cdb6
Packit 98cdb6
#if G_BYTE_ORDER == G_LITTLE_ENDIAN	    
Packit 98cdb6
		if (dest_byteorder == GDK_LSB_FIRST)
Packit 98cdb6
		  {
Packit 98cdb6
		    /* ABGR => ARGB */
Packit 98cdb6
		
Packit 98cdb6
		    while (p < end)
Packit 98cdb6
		      {
Packit 98cdb6
			*q = ( (*p & 0xff00ff00) |
Packit 98cdb6
			       ((*p & 0x000000ff) << 16) |
Packit 98cdb6
			       ((*p & 0x00ff0000) >> 16));
Packit 98cdb6
			q++;
Packit 98cdb6
			p++;
Packit 98cdb6
		      }
Packit 98cdb6
		  }
Packit 98cdb6
		else
Packit 98cdb6
		  {
Packit 98cdb6
		    /* ABGR => BGRA */
Packit 98cdb6
		
Packit 98cdb6
		    while (p < end)
Packit 98cdb6
		      {
Packit 98cdb6
			*q = (((*p & 0xff000000) >> 24) |
Packit 98cdb6
			      ((*p & 0x00ffffff) << 8));
Packit 98cdb6
			q++;
Packit 98cdb6
			p++;
Packit 98cdb6
		      }
Packit 98cdb6
		  }
Packit 98cdb6
#else /* G_BYTE_ORDER == G_BIG_ENDIAN */
Packit 98cdb6
		if (dest_byteorder == GDK_LSB_FIRST)
Packit 98cdb6
		  {
Packit 98cdb6
		    /* RGBA => BGRA */
Packit 98cdb6
		
Packit 98cdb6
		    while (p < end)
Packit 98cdb6
		      {
Packit 98cdb6
			*q = ( (*p & 0x00ff00ff) |
Packit 98cdb6
			       ((*p & 0x0000ff00) << 16) |
Packit 98cdb6
			       ((*p & 0xff000000) >> 16));
Packit 98cdb6
			q++;
Packit 98cdb6
			p++;
Packit 98cdb6
		      }
Packit 98cdb6
		  }
Packit 98cdb6
		else
Packit 98cdb6
		  {
Packit 98cdb6
		    /* RGBA => ARGB */
Packit 98cdb6
		
Packit 98cdb6
		    while (p < end)
Packit 98cdb6
		      {
Packit 98cdb6
			*q = (((*p & 0xffffff00) >> 8) |
Packit 98cdb6
			      ((*p & 0x000000ff) << 24));
Packit 98cdb6
			q++;
Packit 98cdb6
			p++;
Packit 98cdb6
		      }
Packit 98cdb6
		  }
Packit 98cdb6
#endif /* G_BYTE_ORDER*/	    
Packit 98cdb6
	      }
Packit 98cdb6
	    break;
Packit 98cdb6
	  }
Packit 98cdb6
	case GDK_X11_FORMAT_ARGB:
Packit 98cdb6
	  {
Packit 98cdb6
	    guchar *p = (src_buf + i * src_rowstride);
Packit 98cdb6
	    guchar *q = (dest_buf + i * dest_rowstride);
Packit 98cdb6
	    guchar *end = p + 4 * width;
Packit 98cdb6
	    guint t1,t2,t3;
Packit 98cdb6
	    
Packit 98cdb6
#define MULT(d,c,a,t) G_STMT_START { t = c * a; d = ((t >> 8) + t) >> 8; } G_STMT_END
Packit 98cdb6
	    
Packit 98cdb6
	    if (dest_byteorder == GDK_LSB_FIRST)
Packit 98cdb6
	      {
Packit 98cdb6
		while (p < end)
Packit 98cdb6
		  {
Packit 98cdb6
		    MULT(q[0], p[2], p[3], t1);
Packit 98cdb6
		    MULT(q[1], p[1], p[3], t2);
Packit 98cdb6
		    MULT(q[2], p[0], p[3], t3);
Packit 98cdb6
		    q[3] = p[3];
Packit 98cdb6
		    p += 4;
Packit 98cdb6
		    q += 4;
Packit 98cdb6
		  }
Packit 98cdb6
	      }
Packit 98cdb6
	    else
Packit 98cdb6
	      {
Packit 98cdb6
		while (p < end)
Packit 98cdb6
		  {
Packit 98cdb6
		    q[0] = p[3];
Packit 98cdb6
		    MULT(q[1], p[0], p[3], t1);
Packit 98cdb6
		    MULT(q[2], p[1], p[3], t2);
Packit 98cdb6
		    MULT(q[3], p[2], p[3], t3);
Packit 98cdb6
		    p += 4;
Packit 98cdb6
		    q += 4;
Packit 98cdb6
		  }
Packit 98cdb6
	      }
Packit 98cdb6
#undef MULT
Packit 98cdb6
	    break;
Packit 98cdb6
	  }
Packit 98cdb6
	case GDK_X11_FORMAT_NONE:
Packit 98cdb6
	  g_assert_not_reached ();
Packit 98cdb6
	  break;
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
draw_with_images (GdkDrawable       *drawable,
Packit 98cdb6
		  GdkGC             *gc,
Packit 98cdb6
		  GdkX11FormatType   format_type,
Packit 98cdb6
		  XRenderPictFormat *format,
Packit 98cdb6
		  XRenderPictFormat *mask_format,
Packit 98cdb6
		  guchar            *src_rgb,
Packit 98cdb6
		  gint               src_rowstride,
Packit 98cdb6
		  gint               dest_x,
Packit 98cdb6
		  gint               dest_y,
Packit 98cdb6
		  gint               width,
Packit 98cdb6
		  gint               height)
Packit 98cdb6
{
Packit 98cdb6
  GdkScreen *screen = GDK_DRAWABLE_IMPL_X11 (drawable)->screen;
Packit 98cdb6
  Display *xdisplay = GDK_SCREEN_XDISPLAY (screen);
Packit 98cdb6
  GdkImage *image;
Packit 98cdb6
  GdkPixmap *pix;
Packit 98cdb6
  GdkGC *pix_gc;
Packit 98cdb6
  Picture pict;
Packit 98cdb6
  Picture dest_pict;
Packit 98cdb6
  Picture mask = None;
Packit 98cdb6
  gint x0, y0;
Packit 98cdb6
Packit 98cdb6
  pix = gdk_pixmap_new (gdk_screen_get_root_window (screen), width, height, 32);
Packit 98cdb6
						  
Packit 98cdb6
  pict = XRenderCreatePicture (xdisplay, 
Packit 98cdb6
			       GDK_PIXMAP_XID (pix),
Packit 98cdb6
			       format, 0, NULL);
Packit 98cdb6
  if (mask_format)
Packit 98cdb6
    mask = XRenderCreatePicture (xdisplay, 
Packit 98cdb6
				 GDK_PIXMAP_XID (pix),
Packit 98cdb6
				 mask_format, 0, NULL);
Packit 98cdb6
Packit 98cdb6
  dest_pict = gdk_x11_drawable_get_picture (drawable);  
Packit 98cdb6
  
Packit 98cdb6
  pix_gc = _gdk_drawable_get_scratch_gc (pix, FALSE);
Packit 98cdb6
Packit 98cdb6
  for (y0 = 0; y0 < height; y0 += GDK_SCRATCH_IMAGE_HEIGHT)
Packit 98cdb6
    {
Packit 98cdb6
      gint height1 = MIN (height - y0, GDK_SCRATCH_IMAGE_HEIGHT);
Packit 98cdb6
      for (x0 = 0; x0 < width; x0 += GDK_SCRATCH_IMAGE_WIDTH)
Packit 98cdb6
	{
Packit 98cdb6
	  gint xs0, ys0;
Packit 98cdb6
	  
Packit 98cdb6
	  gint width1 = MIN (width - x0, GDK_SCRATCH_IMAGE_WIDTH);
Packit 98cdb6
	  
Packit 98cdb6
	  image = _gdk_image_get_scratch (screen, width1, height1, 32, &xs0, &ys0;;
Packit 98cdb6
	  
Packit 98cdb6
	  _gdk_x11_convert_to_format (src_rgb + y0 * src_rowstride + 4 * x0, src_rowstride,
Packit 98cdb6
                                      (guchar *)image->mem + ys0 * image->bpl + xs0 * image->bpp, image->bpl,
Packit 98cdb6
                                      format_type, image->byte_order, 
Packit 98cdb6
                                      width1, height1);
Packit 98cdb6
Packit 98cdb6
	  gdk_draw_image (pix, pix_gc,
Packit 98cdb6
			  image, xs0, ys0, x0, y0, width1, height1);
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
  
Packit 98cdb6
  XRenderComposite (xdisplay, PictOpOver, pict, mask, dest_pict, 
Packit 98cdb6
		    0, 0, 0, 0, dest_x, dest_y, width, height);
Packit 98cdb6
Packit 98cdb6
  XRenderFreePicture (xdisplay, pict);
Packit 98cdb6
  if (mask)
Packit 98cdb6
    XRenderFreePicture (xdisplay, mask);
Packit 98cdb6
  
Packit 98cdb6
  g_object_unref (pix);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
typedef struct _ShmPixmapInfo ShmPixmapInfo;
Packit 98cdb6
Packit 98cdb6
struct _ShmPixmapInfo
Packit 98cdb6
{
Packit 98cdb6
  Display  *display;
Packit 98cdb6
  Pixmap    pix;
Packit 98cdb6
  Picture   pict;
Packit 98cdb6
  Picture   mask;
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
shm_pixmap_info_destroy (gpointer data)
Packit 98cdb6
{
Packit 98cdb6
  ShmPixmapInfo *info = data;
Packit 98cdb6
Packit 98cdb6
  if (info->pict != None)
Packit 98cdb6
    XRenderFreePicture (info->display, info->pict);
Packit 98cdb6
  if (info->mask != None)
Packit 98cdb6
    XRenderFreePicture (info->display, info->mask);
Packit 98cdb6
Packit 98cdb6
  g_free (data);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
#ifdef USE_SHM
Packit 98cdb6
/* Returns FALSE if we can't get a shm pixmap */
Packit 98cdb6
static gboolean
Packit 98cdb6
get_shm_pixmap_for_image (Display           *xdisplay,
Packit 98cdb6
			  GdkImage          *image,
Packit 98cdb6
			  XRenderPictFormat *format,
Packit 98cdb6
			  XRenderPictFormat *mask_format,
Packit 98cdb6
			  Pixmap            *pix,
Packit 98cdb6
			  Picture           *pict,
Packit 98cdb6
			  Picture           *mask)
Packit 98cdb6
{
Packit 98cdb6
  ShmPixmapInfo *info;
Packit 98cdb6
  
Packit 98cdb6
  if (image->type != GDK_IMAGE_SHARED)
Packit 98cdb6
    return FALSE;
Packit 98cdb6
  
Packit 98cdb6
  info = g_object_get_data (G_OBJECT (image), "gdk-x11-shm-pixmap");
Packit 98cdb6
  if (!info)
Packit 98cdb6
    {
Packit 98cdb6
      *pix = _gdk_x11_image_get_shm_pixmap (image);
Packit 98cdb6
      
Packit 98cdb6
      if (!*pix)
Packit 98cdb6
	return FALSE;
Packit 98cdb6
      
Packit 98cdb6
      info = g_new (ShmPixmapInfo, 1);
Packit 98cdb6
      info->display = xdisplay;
Packit 98cdb6
      info->pix = *pix;
Packit 98cdb6
      
Packit 98cdb6
      info->pict = XRenderCreatePicture (xdisplay, info->pix,
Packit 98cdb6
					 format, 0, NULL);
Packit 98cdb6
      if (mask_format)
Packit 98cdb6
	info->mask = XRenderCreatePicture (xdisplay, info->pix,
Packit 98cdb6
					   mask_format, 0, NULL);
Packit 98cdb6
      else
Packit 98cdb6
	info->mask = None;
Packit 98cdb6
Packit 98cdb6
      g_object_set_data_full (G_OBJECT (image), "gdk-x11-shm-pixmap", info,
Packit 98cdb6
	  shm_pixmap_info_destroy);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  *pix = info->pix;
Packit 98cdb6
  *pict = info->pict;
Packit 98cdb6
  *mask = info->mask;
Packit 98cdb6
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/* Returns FALSE if drawing with ShmPixmaps is not possible */
Packit 98cdb6
static gboolean
Packit 98cdb6
draw_with_pixmaps (GdkDrawable       *drawable,
Packit 98cdb6
		   GdkGC             *gc,
Packit 98cdb6
		   GdkX11FormatType   format_type,
Packit 98cdb6
		   XRenderPictFormat *format,
Packit 98cdb6
		   XRenderPictFormat *mask_format,
Packit 98cdb6
		   guchar            *src_rgb,
Packit 98cdb6
		   gint               src_rowstride,
Packit 98cdb6
		   gint               dest_x,
Packit 98cdb6
		   gint               dest_y,
Packit 98cdb6
		   gint               width,
Packit 98cdb6
		   gint               height)
Packit 98cdb6
{
Packit 98cdb6
  Display *xdisplay = GDK_SCREEN_XDISPLAY (GDK_DRAWABLE_IMPL_X11 (drawable)->screen);
Packit 98cdb6
  GdkImage *image;
Packit 98cdb6
  Pixmap pix;
Packit 98cdb6
  Picture pict;
Packit 98cdb6
  Picture dest_pict;
Packit 98cdb6
  Picture mask = None;
Packit 98cdb6
  gint x0, y0;
Packit 98cdb6
Packit 98cdb6
  dest_pict = gdk_x11_drawable_get_picture (drawable);
Packit 98cdb6
  
Packit 98cdb6
  for (y0 = 0; y0 < height; y0 += GDK_SCRATCH_IMAGE_HEIGHT)
Packit 98cdb6
    {
Packit 98cdb6
      gint height1 = MIN (height - y0, GDK_SCRATCH_IMAGE_HEIGHT);
Packit 98cdb6
      for (x0 = 0; x0 < width; x0 += GDK_SCRATCH_IMAGE_WIDTH)
Packit 98cdb6
	{
Packit 98cdb6
	  gint xs0, ys0;
Packit 98cdb6
	  
Packit 98cdb6
	  gint width1 = MIN (width - x0, GDK_SCRATCH_IMAGE_WIDTH);
Packit 98cdb6
	  
Packit 98cdb6
	  image = _gdk_image_get_scratch (GDK_DRAWABLE_IMPL_X11 (drawable)->screen,
Packit 98cdb6
					  width1, height1, 32, &xs0, &ys0;;
Packit 98cdb6
	  if (!get_shm_pixmap_for_image (xdisplay, image, format, mask_format, &pix, &pict, &mask))
Packit 98cdb6
	    return FALSE;
Packit 98cdb6
Packit 98cdb6
	  _gdk_x11_convert_to_format (src_rgb + y0 * src_rowstride + 4 * x0, src_rowstride,
Packit 98cdb6
                                      (guchar *)image->mem + ys0 * image->bpl + xs0 * image->bpp, image->bpl,
Packit 98cdb6
                                      format_type, image->byte_order, 
Packit 98cdb6
                                      width1, height1);
Packit 98cdb6
Packit 98cdb6
	  XRenderComposite (xdisplay, PictOpOver, pict, mask, dest_pict, 
Packit 98cdb6
			    xs0, ys0, xs0, ys0, x0 + dest_x, y0 + dest_y,
Packit 98cdb6
			    width1, height1);
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6
#endif
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_draw_pixbuf (GdkDrawable     *drawable,
Packit 98cdb6
		     GdkGC           *gc,
Packit 98cdb6
		     GdkPixbuf       *pixbuf,
Packit 98cdb6
		     gint             src_x,
Packit 98cdb6
		     gint             src_y,
Packit 98cdb6
		     gint             dest_x,
Packit 98cdb6
		     gint             dest_y,
Packit 98cdb6
		     gint             width,
Packit 98cdb6
		     gint             height,
Packit 98cdb6
		     GdkRgbDither     dither,
Packit 98cdb6
		     gint             x_dither,
Packit 98cdb6
		     gint             y_dither)
Packit 98cdb6
{
Packit 98cdb6
  GdkX11FormatType format_type;
Packit 98cdb6
  XRenderPictFormat *format, *mask_format;
Packit 98cdb6
  gint rowstride;
Packit 98cdb6
#ifdef USE_SHM  
Packit 98cdb6
  gboolean use_pixmaps = TRUE;
Packit 98cdb6
#endif /* USE_SHM */
Packit 98cdb6
    
Packit 98cdb6
  format_type = select_format (gdk_drawable_get_display (drawable),
Packit 98cdb6
			       &format, &mask_format);
Packit 98cdb6
Packit 98cdb6
  if (format_type == GDK_X11_FORMAT_NONE ||
Packit 98cdb6
      !gdk_pixbuf_get_has_alpha (pixbuf) ||
Packit 98cdb6
      gdk_drawable_get_depth (drawable) == 1 ||
Packit 98cdb6
      (dither == GDK_RGB_DITHER_MAX && gdk_drawable_get_depth (drawable) != 24) ||
Packit 98cdb6
      gdk_x11_drawable_get_picture (drawable) == None)
Packit 98cdb6
    {
Packit 98cdb6
      GdkDrawable *wrapper = GDK_DRAWABLE_IMPL_X11 (drawable)->wrapper;
Packit 98cdb6
      GDK_DRAWABLE_CLASS (_gdk_drawable_impl_x11_parent_class)->draw_pixbuf (wrapper, gc, pixbuf,
Packit 98cdb6
									     src_x, src_y, dest_x, dest_y,
Packit 98cdb6
									     width, height,
Packit 98cdb6
									     dither, x_dither, y_dither);
Packit 98cdb6
      return;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  gdk_x11_drawable_update_picture_clip (drawable, gc);
Packit 98cdb6
Packit 98cdb6
  rowstride = gdk_pixbuf_get_rowstride (pixbuf);
Packit 98cdb6
Packit 98cdb6
#ifdef USE_SHM
Packit 98cdb6
  if (use_pixmaps)
Packit 98cdb6
    {
Packit 98cdb6
      if (!draw_with_pixmaps (drawable, gc,
Packit 98cdb6
			      format_type, format, mask_format,
Packit 98cdb6
			      gdk_pixbuf_get_pixels (pixbuf) + src_y * rowstride + src_x * 4,
Packit 98cdb6
			      rowstride,
Packit 98cdb6
			      dest_x, dest_y, width, height))
Packit 98cdb6
	use_pixmaps = FALSE;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  if (!use_pixmaps)
Packit 98cdb6
#endif /* USE_SHM */
Packit 98cdb6
    draw_with_images (drawable, gc,
Packit 98cdb6
		      format_type, format, mask_format,
Packit 98cdb6
		      gdk_pixbuf_get_pixels (pixbuf) + src_y * rowstride + src_x * 4,
Packit 98cdb6
		      rowstride,
Packit 98cdb6
		      dest_x, dest_y, width, height);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_cairo_surface_destroy (void *data)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl = data;
Packit 98cdb6
Packit 98cdb6
  impl->cairo_surface = NULL;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
_gdk_windowing_set_cairo_surface_size (cairo_surface_t *surface,
Packit 98cdb6
				       int width,
Packit 98cdb6
				       int height)
Packit 98cdb6
{
Packit 98cdb6
  cairo_xlib_surface_set_size (surface, width, height);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
cairo_surface_t *
Packit 98cdb6
_gdk_windowing_create_cairo_surface (GdkDrawable *drawable,
Packit 98cdb6
				     int width,
Packit 98cdb6
				     int height)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
  GdkVisual *visual;
Packit 98cdb6
    
Packit 98cdb6
  visual = gdk_drawable_get_visual (drawable);
Packit 98cdb6
  if (visual) 
Packit 98cdb6
    return cairo_xlib_surface_create (GDK_SCREEN_XDISPLAY (impl->screen),
Packit 98cdb6
				      impl->xid,
Packit 98cdb6
				      GDK_VISUAL_XVISUAL (visual),
Packit 98cdb6
				      width, height);
Packit 98cdb6
  else if (gdk_drawable_get_depth (drawable) == 1)
Packit 98cdb6
    return cairo_xlib_surface_create_for_bitmap (GDK_SCREEN_XDISPLAY (impl->screen),
Packit 98cdb6
						    impl->xid,
Packit 98cdb6
						    GDK_SCREEN_XSCREEN (impl->screen),
Packit 98cdb6
						    width, height);
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      g_warning ("Using Cairo rendering requires the drawable argument to\n"
Packit 98cdb6
		 "have a specified colormap. All windows have a colormap,\n"
Packit 98cdb6
		 "however, pixmaps only have colormap by default if they\n"
Packit 98cdb6
		 "were created with a non-NULL window argument. Otherwise\n"
Packit 98cdb6
		 "a colormap must be set on them with gdk_drawable_set_colormap");
Packit 98cdb6
      return NULL;
Packit 98cdb6
    }
Packit 98cdb6
  
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static cairo_surface_t *
Packit 98cdb6
gdk_x11_ref_cairo_surface (GdkDrawable *drawable)
Packit 98cdb6
{
Packit 98cdb6
  GdkDrawableImplX11 *impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
Packit 98cdb6
  if (GDK_IS_WINDOW_IMPL_X11 (drawable) &&
Packit 98cdb6
      GDK_WINDOW_DESTROYED (impl->wrapper))
Packit 98cdb6
    return NULL;
Packit 98cdb6
Packit 98cdb6
  if (!impl->cairo_surface)
Packit 98cdb6
    {
Packit 98cdb6
      int width, height;
Packit 98cdb6
  
Packit 98cdb6
      gdk_drawable_get_size (impl->wrapper, &width, &height);
Packit 98cdb6
Packit 98cdb6
      impl->cairo_surface = _gdk_windowing_create_cairo_surface (drawable, width, height);
Packit 98cdb6
      
Packit 98cdb6
      if (impl->cairo_surface)
Packit 98cdb6
	cairo_surface_set_user_data (impl->cairo_surface, &gdk_x11_cairo_key,
Packit 98cdb6
				     drawable, gdk_x11_cairo_surface_destroy);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    cairo_surface_reference (impl->cairo_surface);
Packit 98cdb6
Packit 98cdb6
  return impl->cairo_surface;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
#define __GDK_DRAWABLE_X11_C__
Packit 98cdb6
#include "gdkaliasdef.c"