Blame gdk/x11/gdkimage-x11.c

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