Blame gdk/x11/gdkimage-x11.c

Packit 98cdb6
/* GDK - The 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 <stdlib.h>
Packit 98cdb6
#include <sys/types.h>
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 <sys/ipc.h>
Packit 98cdb6
#include <sys/shm.h>
Packit 98cdb6
#endif /* USE_SHM */
Packit 98cdb6
Packit 98cdb6
#include <X11/Xlib.h>
Packit 98cdb6
#include <X11/Xutil.h>
Packit 98cdb6
Packit 98cdb6
#ifdef USE_SHM
Packit 98cdb6
#include <X11/extensions/XShm.h>
Packit 98cdb6
#endif /* USE_SHM */
Packit 98cdb6
Packit 98cdb6
#include <errno.h>
Packit 98cdb6
Packit 98cdb6
#include "gdk.h"		/* For gdk_error_trap_* / gdk_flush_* */
Packit 98cdb6
#include "gdkx.h"
Packit 98cdb6
#include "gdkimage.h"
Packit 98cdb6
#include "gdkprivate.h"
Packit 98cdb6
#include "gdkprivate-x11.h"
Packit 98cdb6
#include "gdkdisplay-x11.h"
Packit 98cdb6
#include "gdkscreen-x11.h"
Packit 98cdb6
#include "gdkalias.h"
Packit 98cdb6
Packit 98cdb6
typedef struct _GdkImagePrivateX11     GdkImagePrivateX11;
Packit 98cdb6
Packit 98cdb6
struct _GdkImagePrivateX11
Packit 98cdb6
{
Packit 98cdb6
  XImage *ximage;
Packit 98cdb6
  GdkScreen *screen;
Packit 98cdb6
  gpointer x_shm_info;
Packit 98cdb6
  Pixmap shm_pixmap;
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
static GList *image_list = NULL;
Packit 98cdb6
Packit 98cdb6
static void gdk_x11_image_destroy (GdkImage      *image);
Packit 98cdb6
static void gdk_image_finalize    (GObject       *object);
Packit 98cdb6
Packit 98cdb6
#define PRIVATE_DATA(image) ((GdkImagePrivateX11 *) GDK_IMAGE (image)->windowing_data)
Packit 98cdb6
Packit 98cdb6
G_DEFINE_TYPE (GdkImage, gdk_image, G_TYPE_OBJECT)
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_image_init (GdkImage *image)
Packit 98cdb6
{
Packit 98cdb6
  image->windowing_data = G_TYPE_INSTANCE_GET_PRIVATE (image, 
Packit 98cdb6
						       GDK_TYPE_IMAGE, 
Packit 98cdb6
						       GdkImagePrivateX11);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_image_class_init (GdkImageClass *klass)
Packit 98cdb6
{
Packit 98cdb6
  GObjectClass *object_class = G_OBJECT_CLASS (klass);
Packit 98cdb6
Packit 98cdb6
  object_class->finalize = gdk_image_finalize;
Packit 98cdb6
Packit 98cdb6
  g_type_class_add_private (object_class, sizeof (GdkImagePrivateX11));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_image_finalize (GObject *object)
Packit 98cdb6
{
Packit 98cdb6
  GdkImage *image = GDK_IMAGE (object);
Packit 98cdb6
Packit 98cdb6
  gdk_x11_image_destroy (image);
Packit 98cdb6
  
Packit 98cdb6
  G_OBJECT_CLASS (gdk_image_parent_class)->finalize (object);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
_gdk_image_exit (void)
Packit 98cdb6
{
Packit 98cdb6
  GdkImage *image;
Packit 98cdb6
Packit 98cdb6
  while (image_list)
Packit 98cdb6
    {
Packit 98cdb6
      image = image_list->data;
Packit 98cdb6
      gdk_x11_image_destroy (image);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gdk_image_new_bitmap:
Packit 98cdb6
 * @visual: the #GdkVisual to use for the image.
Packit 98cdb6
 * @data: the pixel data. 
Packit 98cdb6
 * @width: the width of the image in pixels. 
Packit 98cdb6
 * @height: the height of the image in pixels. 
Packit 98cdb6
 * 
Packit 98cdb6
 * Creates a new #GdkImage with a depth of 1 from the given data.
Packit 98cdb6
 * <warning><para>THIS FUNCTION IS INCREDIBLY BROKEN. The passed-in data must 
Packit 98cdb6
 * be allocated by malloc() (NOT g_malloc()) and will be freed when the 
Packit 98cdb6
 * image is freed.</para></warning>
Packit 98cdb6
 * 
Packit 98cdb6
 * Return value: a new #GdkImage.
Packit 98cdb6
 **/
Packit 98cdb6
GdkImage *
Packit 98cdb6
gdk_image_new_bitmap (GdkVisual *visual, 
Packit 98cdb6
		      gpointer   data, 
Packit 98cdb6
		      gint       width, 
Packit 98cdb6
		      gint       height)
Packit 98cdb6
{
Packit 98cdb6
  Visual *xvisual;
Packit 98cdb6
  GdkImage *image;
Packit 98cdb6
  GdkDisplay *display;
Packit 98cdb6
  GdkImagePrivateX11 *private;
Packit 98cdb6
  
Packit 98cdb6
  image = g_object_new (gdk_image_get_type (), NULL);
Packit 98cdb6
  private = PRIVATE_DATA (image);
Packit 98cdb6
  private->screen = gdk_visual_get_screen (visual);
Packit 98cdb6
  display = GDK_SCREEN_DISPLAY (private->screen);
Packit 98cdb6
  
Packit 98cdb6
  image->type = GDK_IMAGE_NORMAL;
Packit 98cdb6
  image->visual = visual;
Packit 98cdb6
  image->width = width;
Packit 98cdb6
  image->height = height;
Packit 98cdb6
  image->depth = 1;
Packit 98cdb6
  image->bits_per_pixel = 1;
Packit 98cdb6
  if (display->closed)
Packit 98cdb6
    private->ximage = NULL;
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      xvisual = ((GdkVisualPrivate*) visual)->xvisual;
Packit 98cdb6
      private->ximage = XCreateImage (GDK_SCREEN_XDISPLAY (private->screen),
Packit 98cdb6
				      xvisual, 1, XYBitmap,
Packit 98cdb6
				      0, NULL, width, height, 8, 0);
Packit 98cdb6
      private->ximage->data = data;
Packit 98cdb6
      private->ximage->bitmap_bit_order = MSBFirst;
Packit 98cdb6
      private->ximage->byte_order = MSBFirst;
Packit 98cdb6
    }
Packit 98cdb6
  
Packit 98cdb6
  image->byte_order = MSBFirst;
Packit 98cdb6
  image->mem =  private->ximage->data;
Packit 98cdb6
  image->bpl = private->ximage->bytes_per_line;
Packit 98cdb6
  image->bpp = 1;
Packit 98cdb6
  return image;
Packit 98cdb6
} 
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
_gdk_windowing_image_init (GdkDisplay *display)
Packit 98cdb6
{
Packit 98cdb6
  GdkDisplayX11 *display_x11 = GDK_DISPLAY_X11 (display);
Packit 98cdb6
  
Packit 98cdb6
  if (display_x11->use_xshm)
Packit 98cdb6
    {
Packit 98cdb6
#ifdef USE_SHM
Packit 98cdb6
      Display *xdisplay = display_x11->xdisplay;
Packit 98cdb6
      int major, minor, event_base;
Packit 98cdb6
      Bool pixmaps;
Packit 98cdb6
  
Packit 98cdb6
      if (XShmQueryExtension (xdisplay) &&
Packit 98cdb6
	  XShmQueryVersion (xdisplay, &major, &minor, &pixmaps))
Packit 98cdb6
	{
Packit 98cdb6
	  display_x11->have_shm_pixmaps = pixmaps;
Packit 98cdb6
	  event_base = XShmGetEventBase (xdisplay);
Packit 98cdb6
Packit 98cdb6
	  gdk_x11_register_standard_event_type (display,
Packit 98cdb6
						event_base, ShmNumberEvents);
Packit 98cdb6
	}
Packit 98cdb6
      else
Packit 98cdb6
#endif /* USE_SHM */
Packit 98cdb6
	display_x11->use_xshm = FALSE;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
GdkImage*
Packit 98cdb6
_gdk_image_new_for_depth (GdkScreen    *screen,
Packit 98cdb6
			  GdkImageType  type,
Packit 98cdb6
			  GdkVisual    *visual,
Packit 98cdb6
			  gint          width,
Packit 98cdb6
			  gint          height,
Packit 98cdb6
			  gint          depth)
Packit 98cdb6
{
Packit 98cdb6
  GdkImage *image;
Packit 98cdb6
  GdkImagePrivateX11 *private;
Packit 98cdb6
#ifdef USE_SHM
Packit 98cdb6
  XShmSegmentInfo *x_shm_info;
Packit 98cdb6
#endif /* USE_SHM */
Packit 98cdb6
  Visual *xvisual = NULL;
Packit 98cdb6
  GdkDisplayX11 *display_x11;
Packit 98cdb6
  GdkScreenX11 *screen_x11;
Packit 98cdb6
Packit 98cdb6
  g_return_val_if_fail (!visual || GDK_IS_VISUAL (visual), NULL);
Packit 98cdb6
  g_return_val_if_fail (visual || depth != -1, NULL);
Packit 98cdb6
  g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
Packit 98cdb6
  
Packit 98cdb6
  screen_x11 = GDK_SCREEN_X11 (screen);
Packit 98cdb6
  display_x11 = GDK_DISPLAY_X11 (screen_x11->display);
Packit 98cdb6
  
Packit 98cdb6
  if (visual)
Packit 98cdb6
    depth = visual->depth;
Packit 98cdb6
  
Packit 98cdb6
  switch (type)
Packit 98cdb6
    {
Packit 98cdb6
    case GDK_IMAGE_FASTEST:
Packit 98cdb6
      image = _gdk_image_new_for_depth (screen, GDK_IMAGE_SHARED, 
Packit 98cdb6
					visual, width, height, depth);
Packit 98cdb6
      if (!image)
Packit 98cdb6
	image = _gdk_image_new_for_depth (screen, GDK_IMAGE_NORMAL,
Packit 98cdb6
					  visual, width, height, depth);
Packit 98cdb6
      break;
Packit 98cdb6
Packit 98cdb6
    default:
Packit 98cdb6
      image = g_object_new (gdk_image_get_type (), NULL);
Packit 98cdb6
      
Packit 98cdb6
      private = PRIVATE_DATA (image);
Packit 98cdb6
Packit 98cdb6
      private->screen = screen;
Packit 98cdb6
Packit 98cdb6
      image->type = type;
Packit 98cdb6
      image->visual = visual;
Packit 98cdb6
      image->width = width;
Packit 98cdb6
      image->height = height;
Packit 98cdb6
      image->depth = depth;
Packit 98cdb6
Packit 98cdb6
      if (visual)
Packit 98cdb6
	xvisual = ((GdkVisualPrivate*) visual)->xvisual;
Packit 98cdb6
Packit 98cdb6
      switch (type)
Packit 98cdb6
	{
Packit 98cdb6
	case GDK_IMAGE_SHARED:
Packit 98cdb6
#ifdef USE_SHM
Packit 98cdb6
	  if (display_x11->use_xshm)
Packit 98cdb6
	    {
Packit 98cdb6
	      private->x_shm_info = g_new (XShmSegmentInfo, 1);
Packit 98cdb6
	      x_shm_info = private->x_shm_info;
Packit 98cdb6
	      x_shm_info->shmid = -1;
Packit 98cdb6
	      x_shm_info->shmaddr = (char*) -1;
Packit 98cdb6
Packit 98cdb6
	      private->ximage = XShmCreateImage (screen_x11->xdisplay, xvisual, depth,
Packit 98cdb6
						 ZPixmap, NULL, x_shm_info, width, height);
Packit 98cdb6
	      if (private->ximage == NULL)
Packit 98cdb6
		{
Packit 98cdb6
		  g_warning ("XShmCreateImage failed");
Packit 98cdb6
		  display_x11->use_xshm = FALSE;
Packit 98cdb6
		  
Packit 98cdb6
		  goto error;
Packit 98cdb6
		}
Packit 98cdb6
Packit 98cdb6
	      x_shm_info->shmid = shmget (IPC_PRIVATE,
Packit 98cdb6
					  private->ximage->bytes_per_line * private->ximage->height,
Packit 98cdb6
					  IPC_CREAT | 0600);
Packit 98cdb6
Packit 98cdb6
	      if (x_shm_info->shmid == -1)
Packit 98cdb6
		{
Packit 98cdb6
		  /* EINVAL indicates, most likely, that the segment we asked for
Packit 98cdb6
		   * is bigger than SHMMAX, so we don't treat it as a permanent
Packit 98cdb6
		   * error. ENOSPC and ENOMEM may also indicate this, but
Packit 98cdb6
		   * more likely are permanent errors.
Packit 98cdb6
		   */
Packit 98cdb6
		  if (errno != EINVAL)
Packit 98cdb6
		    {
Packit 98cdb6
		      g_warning ("shmget failed: error %d (%s)", errno, g_strerror (errno));
Packit 98cdb6
		      display_x11->use_xshm = FALSE;
Packit 98cdb6
		    }
Packit 98cdb6
Packit 98cdb6
		  goto error;
Packit 98cdb6
		}
Packit 98cdb6
Packit 98cdb6
	      x_shm_info->readOnly = False;
Packit 98cdb6
	      x_shm_info->shmaddr = shmat (x_shm_info->shmid, NULL, 0);
Packit 98cdb6
	      private->ximage->data = x_shm_info->shmaddr;
Packit 98cdb6
Packit 98cdb6
	      if (x_shm_info->shmaddr == (char*) -1)
Packit 98cdb6
		{
Packit 98cdb6
		  g_warning ("shmat failed: error %d (%s)", errno, g_strerror (errno));
Packit 98cdb6
		  /* Failure in shmat is almost certainly permanent. Most likely error is
Packit 98cdb6
		   * EMFILE, which would mean that we've exceeded the per-process
Packit 98cdb6
		   * Shm segment limit.
Packit 98cdb6
		   */
Packit 98cdb6
		  display_x11->use_xshm = FALSE;
Packit 98cdb6
		  goto error;
Packit 98cdb6
		}
Packit 98cdb6
Packit 98cdb6
	      gdk_error_trap_push ();
Packit 98cdb6
Packit 98cdb6
	      XShmAttach (screen_x11->xdisplay, x_shm_info);
Packit 98cdb6
	      XSync (screen_x11->xdisplay, False);
Packit 98cdb6
Packit 98cdb6
	      if (gdk_error_trap_pop ())
Packit 98cdb6
		{
Packit 98cdb6
		  /* this is the common failure case so omit warning */
Packit 98cdb6
		  display_x11->use_xshm = FALSE;
Packit 98cdb6
		  goto error;
Packit 98cdb6
		}
Packit 98cdb6
	      
Packit 98cdb6
	      /* We mark the segment as destroyed so that when
Packit 98cdb6
	       * the last process detaches, it will be deleted.
Packit 98cdb6
	       * There is a small possibility of leaking if
Packit 98cdb6
	       * we die in XShmAttach. In theory, a signal handler
Packit 98cdb6
	       * could be set up.
Packit 98cdb6
	       */
Packit 98cdb6
	      shmctl (x_shm_info->shmid, IPC_RMID, NULL);		      
Packit 98cdb6
Packit 98cdb6
	      if (image)
Packit 98cdb6
		image_list = g_list_prepend (image_list, image);
Packit 98cdb6
	    }
Packit 98cdb6
	  else
Packit 98cdb6
#endif /* USE_SHM */
Packit 98cdb6
	    goto error;
Packit 98cdb6
	  break;
Packit 98cdb6
	case GDK_IMAGE_NORMAL:
Packit 98cdb6
	  private->ximage = XCreateImage (screen_x11->xdisplay, xvisual, depth,
Packit 98cdb6
					  ZPixmap, 0, NULL, width, height, 32, 0);
Packit 98cdb6
Packit 98cdb6
	  /* Use malloc, not g_malloc here, because X will call free()
Packit 98cdb6
	   * on this data
Packit 98cdb6
	   */
Packit 98cdb6
	  private->ximage->data = malloc (private->ximage->bytes_per_line *
Packit 98cdb6
					  private->ximage->height);
Packit 98cdb6
	  if (!private->ximage->data)
Packit 98cdb6
	    goto error;
Packit 98cdb6
	  break;
Packit 98cdb6
Packit 98cdb6
	case GDK_IMAGE_FASTEST:
Packit 98cdb6
	  g_assert_not_reached ();
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      if (image)
Packit 98cdb6
	{
Packit 98cdb6
	  image->byte_order = (private->ximage->byte_order == LSBFirst) ? GDK_LSB_FIRST : GDK_MSB_FIRST;
Packit 98cdb6
	  image->mem = private->ximage->data;
Packit 98cdb6
	  image->bpl = private->ximage->bytes_per_line;
Packit 98cdb6
	  image->bpp = (private->ximage->bits_per_pixel + 7) / 8;
Packit 98cdb6
	  image->bits_per_pixel = private->ximage->bits_per_pixel;
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return image;
Packit 98cdb6
Packit 98cdb6
 error:
Packit 98cdb6
  if (private->ximage)
Packit 98cdb6
    {
Packit 98cdb6
      XDestroyImage (private->ximage);
Packit 98cdb6
      private->ximage = NULL;
Packit 98cdb6
    }
Packit 98cdb6
#ifdef USE_SHM
Packit 98cdb6
  if (private->x_shm_info)
Packit 98cdb6
    {
Packit 98cdb6
      x_shm_info = private->x_shm_info;
Packit 98cdb6
      
Packit 98cdb6
      if (x_shm_info->shmaddr != (char *)-1)
Packit 98cdb6
	shmdt (x_shm_info->shmaddr);
Packit 98cdb6
      if (x_shm_info->shmid != -1) 
Packit 98cdb6
	shmctl (x_shm_info->shmid, IPC_RMID, NULL);
Packit 98cdb6
      
Packit 98cdb6
      g_free (x_shm_info);
Packit 98cdb6
      private->x_shm_info = NULL;
Packit 98cdb6
    }
Packit 98cdb6
#endif /* USE_SHM */
Packit 98cdb6
  g_object_unref (image);
Packit 98cdb6
  
Packit 98cdb6
  return NULL;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Pixmap
Packit 98cdb6
_gdk_x11_image_get_shm_pixmap (GdkImage *image)
Packit 98cdb6
{
Packit 98cdb6
  GdkImagePrivateX11 *private = PRIVATE_DATA (image);
Packit 98cdb6
  GdkDisplay *display = GDK_SCREEN_DISPLAY (private->screen);
Packit 98cdb6
Packit 98cdb6
  if (display->closed)
Packit 98cdb6
    return None;
Packit 98cdb6
Packit 98cdb6
#ifdef USE_SHM  
Packit 98cdb6
  /* Future: do we need one of these per-screen per-image? ShmPixmaps
Packit 98cdb6
   * are the same for every screen, but can they be shared? Not a concern
Packit 98cdb6
   * right now since we tie images to a particular screen.
Packit 98cdb6
   */
Packit 98cdb6
  if (!private->shm_pixmap && image->type == GDK_IMAGE_SHARED && 
Packit 98cdb6
      GDK_DISPLAY_X11 (display)->have_shm_pixmaps)
Packit 98cdb6
    private->shm_pixmap = XShmCreatePixmap (GDK_SCREEN_XDISPLAY (private->screen),
Packit 98cdb6
					    GDK_SCREEN_XROOTWIN (private->screen),
Packit 98cdb6
					    image->mem, private->x_shm_info, 
Packit 98cdb6
					    image->width, image->height, image->depth);
Packit 98cdb6
Packit 98cdb6
  return private->shm_pixmap;
Packit 98cdb6
#else
Packit 98cdb6
  return None;
Packit 98cdb6
#endif    
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static GdkImage*
Packit 98cdb6
get_full_image (GdkDrawable    *drawable,
Packit 98cdb6
		gint            src_x,
Packit 98cdb6
		gint            src_y,
Packit 98cdb6
		gint            width,
Packit 98cdb6
		gint            height)
Packit 98cdb6
{
Packit 98cdb6
  GdkImage *image;
Packit 98cdb6
  GdkImagePrivateX11 *private;
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
  XImage *ximage;
Packit 98cdb6
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
  
Packit 98cdb6
  ximage = XGetImage (GDK_SCREEN_XDISPLAY (impl->screen),
Packit 98cdb6
		      impl->xid,
Packit 98cdb6
		      src_x, src_y, width, height,
Packit 98cdb6
		      AllPlanes, ZPixmap);
Packit 98cdb6
  
Packit 98cdb6
  if (!ximage)
Packit 98cdb6
    return NULL;
Packit 98cdb6
  
Packit 98cdb6
  image = g_object_new (gdk_image_get_type (), NULL);
Packit 98cdb6
  
Packit 98cdb6
  private = PRIVATE_DATA (image);
Packit 98cdb6
  
Packit 98cdb6
  private->screen = impl->screen;
Packit 98cdb6
  private->ximage = ximage;
Packit 98cdb6
  
Packit 98cdb6
  image->type = GDK_IMAGE_NORMAL;
Packit 98cdb6
  image->visual = gdk_drawable_get_visual (drawable); /* could be NULL */
Packit 98cdb6
  image->width = width;
Packit 98cdb6
  image->height = height;
Packit 98cdb6
  image->depth = gdk_drawable_get_depth (drawable);
Packit 98cdb6
  
Packit 98cdb6
  image->mem = private->ximage->data;
Packit 98cdb6
  image->bpl = private->ximage->bytes_per_line;
Packit 98cdb6
  image->bits_per_pixel = private->ximage->bits_per_pixel;
Packit 98cdb6
  image->bpp = (private->ximage->bits_per_pixel + 7) / 8;
Packit 98cdb6
  image->byte_order = (private->ximage->byte_order == LSBFirst) ? GDK_LSB_FIRST : GDK_MSB_FIRST;
Packit 98cdb6
  
Packit 98cdb6
  return image;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
GdkImage*
Packit 98cdb6
_gdk_x11_copy_to_image (GdkDrawable    *drawable,
Packit 98cdb6
			GdkImage       *image,
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
{
Packit 98cdb6
  GdkImagePrivateX11 *private;
Packit 98cdb6
  GdkDrawableImplX11 *impl;
Packit 98cdb6
  GdkVisual *visual;
Packit 98cdb6
  GdkDisplay *display;
Packit 98cdb6
  Display *xdisplay;
Packit 98cdb6
  gboolean have_grab;
Packit 98cdb6
  GdkRectangle req;
Packit 98cdb6
  GdkRectangle window_rect;
Packit 98cdb6
  Pixmap shm_pixmap = None;
Packit 98cdb6
  gboolean success = TRUE;
Packit 98cdb6
  
Packit 98cdb6
  g_return_val_if_fail (GDK_IS_DRAWABLE_IMPL_X11 (drawable), NULL);
Packit 98cdb6
  g_return_val_if_fail (image != NULL || (dest_x == 0 && dest_y == 0), NULL);
Packit 98cdb6
Packit 98cdb6
  visual = gdk_drawable_get_visual (drawable);
Packit 98cdb6
  impl = GDK_DRAWABLE_IMPL_X11 (drawable);
Packit 98cdb6
  display = gdk_drawable_get_display (drawable);
Packit 98cdb6
  xdisplay = gdk_x11_display_get_xdisplay (display);
Packit 98cdb6
Packit 98cdb6
  if (display->closed)
Packit 98cdb6
    return NULL;
Packit 98cdb6
  
Packit 98cdb6
  have_grab = FALSE;
Packit 98cdb6
Packit 98cdb6
#define UNGRAB() G_STMT_START {					\
Packit 98cdb6
    if (have_grab) {						\
Packit 98cdb6
      gdk_x11_display_ungrab (display);				\
Packit 98cdb6
      have_grab = FALSE; }					\
Packit 98cdb6
  } G_STMT_END
Packit 98cdb6
Packit 98cdb6
  if (!image && !GDK_IS_WINDOW_IMPL_X11 (drawable))
Packit 98cdb6
    return get_full_image (drawable, src_x, src_y, width, height);
Packit 98cdb6
Packit 98cdb6
  if (image && image->type == GDK_IMAGE_SHARED)
Packit 98cdb6
    {
Packit 98cdb6
      shm_pixmap = _gdk_x11_image_get_shm_pixmap (image);
Packit 98cdb6
      if (shm_pixmap)
Packit 98cdb6
	{
Packit 98cdb6
	  GC xgc;
Packit 98cdb6
	  XGCValues values;
Packit 98cdb6
Packit 98cdb6
	  /* Again easy, we can just XCopyArea, and don't have to worry about clipping
Packit 98cdb6
	   */
Packit 98cdb6
	  values.subwindow_mode = IncludeInferiors;
Packit 98cdb6
	  xgc = XCreateGC (xdisplay, impl->xid, GCSubwindowMode, &values);
Packit 98cdb6
	  
Packit 98cdb6
	  XCopyArea (xdisplay, impl->xid, shm_pixmap, xgc,
Packit 98cdb6
		     src_x, src_y, width, height, dest_x, dest_y);
Packit 98cdb6
	  XSync (xdisplay, FALSE);
Packit 98cdb6
	  
Packit 98cdb6
	  XFreeGC (xdisplay, xgc);
Packit 98cdb6
	  
Packit 98cdb6
	  return image;
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  /* Now the general case - we may have to worry about clipping to the screen
Packit 98cdb6
   * bounds, in which case we'll have to grab the server and only get a piece
Packit 98cdb6
   * of the window.
Packit 98cdb6
   */
Packit 98cdb6
  if (GDK_IS_WINDOW_IMPL_X11 (drawable))
Packit 98cdb6
    {
Packit 98cdb6
      GdkRectangle screen_rect;
Packit 98cdb6
      Window child;
Packit 98cdb6
Packit 98cdb6
      have_grab = TRUE;
Packit 98cdb6
      gdk_x11_display_grab (display);
Packit 98cdb6
Packit 98cdb6
      /* Translate screen area into window coordinates */
Packit 98cdb6
      XTranslateCoordinates (xdisplay,
Packit 98cdb6
			     GDK_SCREEN_XROOTWIN (impl->screen),
Packit 98cdb6
			     impl->xid,
Packit 98cdb6
			     0, 0, 
Packit 98cdb6
			     &screen_rect.x, &screen_rect.y, 
Packit 98cdb6
			     &child);
Packit 98cdb6
Packit 98cdb6
      screen_rect.width = gdk_screen_get_width (impl->screen);
Packit 98cdb6
      screen_rect.height = gdk_screen_get_height (impl->screen);
Packit 98cdb6
      
Packit 98cdb6
      gdk_error_trap_push ();
Packit 98cdb6
Packit 98cdb6
      window_rect.x = 0;
Packit 98cdb6
      window_rect.y = 0;
Packit 98cdb6
      
Packit 98cdb6
      gdk_window_get_geometry (GDK_WINDOW (impl->wrapper),
Packit 98cdb6
                               NULL, NULL,
Packit 98cdb6
                               &window_rect.width,
Packit 98cdb6
                               &window_rect.height,
Packit 98cdb6
                               NULL);
Packit 98cdb6
      
Packit 98cdb6
      /* compute intersection of screen and window, in window
Packit 98cdb6
       * coordinates
Packit 98cdb6
       */
Packit 98cdb6
      if (gdk_error_trap_pop () ||
Packit 98cdb6
          !gdk_rectangle_intersect (&window_rect, &screen_rect, 
Packit 98cdb6
                                    &window_rect))
Packit 98cdb6
	goto out;
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      window_rect.x = 0;
Packit 98cdb6
      window_rect.y = 0;
Packit 98cdb6
      gdk_drawable_get_size (drawable,
Packit 98cdb6
			     &window_rect.width,
Packit 98cdb6
			     &window_rect.height);
Packit 98cdb6
    }
Packit 98cdb6
      
Packit 98cdb6
  req.x = src_x;
Packit 98cdb6
  req.y = src_y;
Packit 98cdb6
  req.width = width;
Packit 98cdb6
  req.height = height;
Packit 98cdb6
  
Packit 98cdb6
  /* window_rect specifies the part of drawable which we can get from
Packit 98cdb6
   * the server in window coordinates. 
Packit 98cdb6
   * For pixmaps this is all of the pixmap, for windows it is just 
Packit 98cdb6
   * the onscreen part.
Packit 98cdb6
   */
Packit 98cdb6
  if (!gdk_rectangle_intersect (&req, &window_rect, &req))
Packit 98cdb6
    goto out;
Packit 98cdb6
Packit 98cdb6
  gdk_error_trap_push ();
Packit 98cdb6
  
Packit 98cdb6
  if (!image &&
Packit 98cdb6
      req.x == src_x && req.y == src_y && req.width == width && req.height == height)
Packit 98cdb6
    {
Packit 98cdb6
      image = get_full_image (drawable, src_x, src_y, width, height);
Packit 98cdb6
      if (!image)
Packit 98cdb6
	success = FALSE;
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      gboolean created_image = FALSE;
Packit 98cdb6
      
Packit 98cdb6
      if (!image)
Packit 98cdb6
	{
Packit 98cdb6
	  image = _gdk_image_new_for_depth (impl->screen, GDK_IMAGE_NORMAL, 
Packit 98cdb6
					    visual, width, height,
Packit 98cdb6
					    gdk_drawable_get_depth (drawable));
Packit 98cdb6
	  created_image = TRUE;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      private = PRIVATE_DATA (image);
Packit 98cdb6
Packit 98cdb6
      /* In the ShmImage but no ShmPixmap case, we could use XShmGetImage when
Packit 98cdb6
       * we are getting the entire image.
Packit 98cdb6
       */
Packit 98cdb6
      if (XGetSubImage (xdisplay, impl->xid,
Packit 98cdb6
			req.x, req.y, req.width, req.height,
Packit 98cdb6
			AllPlanes, ZPixmap,
Packit 98cdb6
			private->ximage,
Packit 98cdb6
			dest_x + req.x - src_x, dest_y + req.y - src_y) == None)
Packit 98cdb6
	{
Packit 98cdb6
	  if (created_image)
Packit 98cdb6
	    g_object_unref (image);
Packit 98cdb6
	  image = NULL;
Packit 98cdb6
	  success = FALSE;
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  gdk_error_trap_pop ();
Packit 98cdb6
Packit 98cdb6
 out:
Packit 98cdb6
  
Packit 98cdb6
  if (have_grab)
Packit 98cdb6
    {				
Packit 98cdb6
      gdk_x11_display_ungrab (display);
Packit 98cdb6
      have_grab = FALSE;
Packit 98cdb6
    }
Packit 98cdb6
  
Packit 98cdb6
  if (success && !image)
Packit 98cdb6
    {
Packit 98cdb6
      /* We "succeeded", but could get no content for the image so return junk */
Packit 98cdb6
      image = _gdk_image_new_for_depth (impl->screen, GDK_IMAGE_NORMAL, 
Packit 98cdb6
					visual, width, height,
Packit 98cdb6
					gdk_drawable_get_depth (drawable));
Packit 98cdb6
    }
Packit 98cdb6
      
Packit 98cdb6
  return image;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
guint32
Packit 98cdb6
gdk_image_get_pixel (GdkImage *image,
Packit 98cdb6
		     gint x,
Packit 98cdb6
		     gint y)
Packit 98cdb6
{
Packit 98cdb6
  guint32 pixel;
Packit 98cdb6
  GdkImagePrivateX11 *private;
Packit 98cdb6
Packit 98cdb6
  g_return_val_if_fail (GDK_IS_IMAGE (image), 0);
Packit 98cdb6
  g_return_val_if_fail (x >= 0 && x < image->width, 0);
Packit 98cdb6
  g_return_val_if_fail (y >= 0 && y < image->height, 0);
Packit 98cdb6
Packit 98cdb6
  private = PRIVATE_DATA (image);
Packit 98cdb6
Packit 98cdb6
  if (!private->screen->closed)
Packit 98cdb6
    pixel = XGetPixel (private->ximage, x, y);
Packit 98cdb6
  else
Packit 98cdb6
    pixel = 0;
Packit 98cdb6
Packit 98cdb6
  return pixel;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
gdk_image_put_pixel (GdkImage *image,
Packit 98cdb6
		     gint x,
Packit 98cdb6
		     gint y,
Packit 98cdb6
		     guint32 pixel)
Packit 98cdb6
{
Packit 98cdb6
  GdkImagePrivateX11 *private;
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (GDK_IS_IMAGE (image));
Packit 98cdb6
  g_return_if_fail (x >= 0 && x < image->width);
Packit 98cdb6
  g_return_if_fail (y >= 0 && y < image->height);
Packit 98cdb6
Packit 98cdb6
  private = PRIVATE_DATA (image);
Packit 98cdb6
Packit 98cdb6
  if (!private->screen->closed)
Packit 98cdb6
    pixel = XPutPixel (private->ximage, x, y, pixel);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_x11_image_destroy (GdkImage *image)
Packit 98cdb6
{
Packit 98cdb6
  GdkImagePrivateX11 *private;
Packit 98cdb6
#ifdef USE_SHM
Packit 98cdb6
  XShmSegmentInfo *x_shm_info;
Packit 98cdb6
#endif /* USE_SHM */
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (GDK_IS_IMAGE (image));
Packit 98cdb6
Packit 98cdb6
  private = PRIVATE_DATA (image);
Packit 98cdb6
Packit 98cdb6
  if (private->ximage)		/* Deal with failure of creation */
Packit 98cdb6
    {
Packit 98cdb6
      switch (image->type)
Packit 98cdb6
	{
Packit 98cdb6
	case GDK_IMAGE_NORMAL:
Packit 98cdb6
	  if (!private->screen->closed)
Packit 98cdb6
	    XDestroyImage (private->ximage);
Packit 98cdb6
	  break;
Packit 98cdb6
	  
Packit 98cdb6
	case GDK_IMAGE_SHARED:
Packit 98cdb6
#ifdef USE_SHM
Packit 98cdb6
	  if (!private->screen->closed)
Packit 98cdb6
	    {
Packit 98cdb6
	      gdk_display_sync (GDK_SCREEN_DISPLAY (private->screen));
Packit 98cdb6
Packit 98cdb6
	      if (private->shm_pixmap)
Packit 98cdb6
		XFreePixmap (GDK_SCREEN_XDISPLAY (private->screen), private->shm_pixmap);
Packit 98cdb6
	  	  
Packit 98cdb6
	      XShmDetach (GDK_SCREEN_XDISPLAY (private->screen), private->x_shm_info);
Packit 98cdb6
	      XDestroyImage (private->ximage);
Packit 98cdb6
	    }
Packit 98cdb6
	  
Packit 98cdb6
	  image_list = g_list_remove (image_list, image);
Packit 98cdb6
Packit 98cdb6
	  x_shm_info = private->x_shm_info;
Packit 98cdb6
	  shmdt (x_shm_info->shmaddr);
Packit 98cdb6
	  
Packit 98cdb6
	  g_free (private->x_shm_info);
Packit 98cdb6
	  private->x_shm_info = NULL;
Packit 98cdb6
Packit 98cdb6
#else /* USE_SHM */
Packit 98cdb6
	  g_error ("trying to destroy shared memory image when gdk was compiled without shared memory support");
Packit 98cdb6
#endif /* USE_SHM */
Packit 98cdb6
	  break;
Packit 98cdb6
	  
Packit 98cdb6
	case GDK_IMAGE_FASTEST:
Packit 98cdb6
	  g_assert_not_reached ();
Packit 98cdb6
	}
Packit 98cdb6
      
Packit 98cdb6
      private->ximage = NULL;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gdk_x11_image_get_xdisplay:
Packit 98cdb6
 * @image: a #GdkImage.
Packit 98cdb6
 * 
Packit 98cdb6
 * Returns the display of a #GdkImage.
Packit 98cdb6
 * 
Packit 98cdb6
 * Return value: an Xlib <type>Display*</type>.
Packit 98cdb6
 **/
Packit 98cdb6
Display *
Packit 98cdb6
gdk_x11_image_get_xdisplay (GdkImage *image)
Packit 98cdb6
{
Packit 98cdb6
  GdkImagePrivateX11 *private;
Packit 98cdb6
Packit 98cdb6
  g_return_val_if_fail (GDK_IS_IMAGE (image), NULL);
Packit 98cdb6
Packit 98cdb6
  private = PRIVATE_DATA (image);
Packit 98cdb6
Packit 98cdb6
  return GDK_SCREEN_XDISPLAY (private->screen);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gdk_x11_image_get_ximage:
Packit 98cdb6
 * @image: a #GdkImage.
Packit 98cdb6
 * 
Packit 98cdb6
 * Returns the X image belonging to a #GdkImage.
Packit 98cdb6
 * 
Packit 98cdb6
 * Return value: an <type>XImage*</type>.
Packit 98cdb6
 **/
Packit 98cdb6
XImage *
Packit 98cdb6
gdk_x11_image_get_ximage (GdkImage *image)
Packit 98cdb6
{
Packit 98cdb6
  GdkImagePrivateX11 *private;
Packit 98cdb6
Packit 98cdb6
  g_return_val_if_fail (GDK_IS_IMAGE (image), NULL);
Packit 98cdb6
Packit 98cdb6
  private = PRIVATE_DATA (image);
Packit 98cdb6
Packit 98cdb6
  if (private->screen->closed)
Packit 98cdb6
    return NULL;
Packit 98cdb6
  else
Packit 98cdb6
    return private->ximage;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
gint
Packit 98cdb6
_gdk_windowing_get_bits_for_depth (GdkDisplay *display,
Packit 98cdb6
				   gint        depth)
Packit 98cdb6
{
Packit 98cdb6
  XPixmapFormatValues *formats;
Packit 98cdb6
  gint count, i;
Packit 98cdb6
Packit 98cdb6
  formats = XListPixmapFormats (GDK_DISPLAY_XDISPLAY (display), &count);
Packit 98cdb6
  
Packit 98cdb6
  for (i = 0; i < count; i++)
Packit 98cdb6
    if (formats[i].depth == depth)
Packit 98cdb6
      {
Packit 98cdb6
	gint result = formats[i].bits_per_pixel;
Packit 98cdb6
	XFree (formats);
Packit 98cdb6
	return result;
Packit 98cdb6
      }
Packit 98cdb6
Packit 98cdb6
  g_assert_not_reached ();
Packit 98cdb6
  return -1;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
#define __GDK_IMAGE_X11_C__
Packit 98cdb6
#include "gdkaliasdef.c"