]> Pileus Git - ~andy/gtk/blob - gdk/gdkrgb.c
Fix #99593: Fix a memory leak when XmbLookupString returns XBufferOverflow
[~andy/gtk] / gdk / gdkrgb.c
1 /* GDK - The GIMP Drawing Kit
2  * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Lesser General Public
6  * License as published by the Free Software Foundation; either
7  * version 2 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Lesser General Public License for more details.
13  *
14  * You should have received a copy of the GNU Lesser General Public
15  * License along with this library; if not, write to the
16  * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17  * Boston, MA 02111-1307, USA.
18  */
19
20 /* For more information on GdkRgb, see http://www.levien.com/gdkrgb/
21
22    Raph Levien <raph@acm.org>
23    */
24
25 /*
26  * Modified by the GTK+ Team and others 1997-2000.  See the AUTHORS
27  * file for a list of people on the GTK+ Team.  See the ChangeLog
28  * files for a list of changes.  These files are distributed with
29  * GTK+ at ftp://ftp.gtk.org/pub/gtk/. 
30  */
31
32 #include <math.h>
33
34 #if HAVE_CONFIG_H
35 #  include <config.h>
36 #  if STDC_HEADERS
37 #    include <stdio.h>
38 #    include <stdlib.h>
39 #    include <string.h>
40 #  endif
41 #else
42 #  include <stdio.h>
43 #  include <stdlib.h>
44 #endif
45
46
47 #define ENABLE_GRAYSCALE
48
49 #include "config.h"
50 #include "gdkprivate.h"
51 #include "gdkinternals.h"       /* _gdk_windowing_get_bits_for_depth() */
52
53 #include "gdkrgb.h"
54 #include "gdkscreen.h"
55
56 typedef struct _GdkRgbInfo     GdkRgbInfo;
57 typedef struct _GdkRgbCmapInfo GdkRgbCmapInfo;
58
59 typedef void (*GdkRgbConvFunc) (GdkRgbInfo *image_info, GdkImage *image,
60                                 gint x0, gint y0,
61                                 gint width, gint height,
62                                 guchar *buf, int rowstride,
63                                 gint x_align, gint y_align,
64                                 GdkRgbCmap *cmap);
65
66 static const gchar* visual_names[] =
67 {
68   "static gray",
69   "grayscale",
70   "static color",
71   "pseudo color",
72   "true color",
73   "direct color",
74 };
75
76 #define STAGE_ROWSTRIDE (GDK_SCRATCH_IMAGE_WIDTH * 3)
77
78 /* Some of these fields should go, as they're not being used at all. (?)
79  */
80 struct _GdkRgbInfo
81 {
82   GdkVisual *visual;
83   GdkColormap *cmap;
84
85   guint nred_shades;
86   guint ngreen_shades;
87   guint nblue_shades;
88   guint ngray_shades;
89   guint nreserved;
90
91   guint bpp;
92   gint cmap_alloced;
93   gdouble gamma;
94
95   /* Generally, the stage buffer is used to convert 32bit RGB, gray,
96      and indexed images into 24 bit packed RGB. */
97   guchar *stage_buf;
98
99   GdkRgbCmap *gray_cmap;
100
101   gboolean dith_default;
102
103   gboolean bitmap; /* set true if in 1 bit per pixel mode */
104   GdkGC *own_gc;
105
106   /* Convert functions */
107   GdkRgbConvFunc conv;
108   GdkRgbConvFunc conv_d;
109
110   GdkRgbConvFunc conv_32;
111   GdkRgbConvFunc conv_32_d;
112
113   GdkRgbConvFunc conv_gray;
114   GdkRgbConvFunc conv_gray_d;
115
116   GdkRgbConvFunc conv_indexed;
117   GdkRgbConvFunc conv_indexed_d;
118
119   guchar *colorcube;
120   guchar *colorcube_d;
121
122   /* We need to track LUT's for pairs of GdkRgbInfo / GdkRgbCmap, so we
123    * keep a list of pointers to GdkRgbCmapInfo on both structures so we
124    * can remove as necessary when freeing a GdkRgbInfo or GdkRgbCmap
125    */
126   GSList *cmap_info_list;
127 };
128
129 struct _GdkRgbCmapInfo
130 {
131   GdkRgbInfo *image_info;
132   GdkRgbCmap *cmap;
133
134   guchar lut[256];              /* For 8-bit modes */
135 };
136
137 static GdkRgbCmapInfo *gdk_rgb_cmap_get_info (GdkRgbCmap *cmap, GdkRgbInfo *image_info);
138
139 static const char *gdk_rgb_key = "gdk-rgb-info";
140 static GQuark gdk_rgb_quark = 0;
141
142 static gboolean gdk_rgb_install_cmap = FALSE;
143 static gint gdk_rgb_min_colors = 5 * 5 * 5;
144 static gboolean gdk_rgb_verbose = FALSE;
145
146 static gint
147 gdk_rgb_cmap_fail (const char *msg, GdkColormap *cmap, gulong *pixels)
148 {
149   GdkColor free_colors[256];
150   gint n_free;
151   gint i;
152
153 #ifdef VERBOSE
154   g_print ("%s", msg);
155 #endif
156   n_free = 0;
157   for (i = 0; i < 256; i++)
158     if (pixels[i] < 256)
159       free_colors[n_free++].pixel = pixels[i];
160   if (n_free)
161     gdk_colormap_free_colors (cmap, free_colors, n_free);
162   return 0;
163 }
164
165 static void
166 gdk_rgb_make_colorcube (GdkRgbInfo *image_info, gulong *pixels,
167                         gint nr, gint ng, gint nb)
168 {
169   guchar rt[16], gt[16], bt[16];
170   gint i;
171
172   image_info->colorcube = g_new (guchar, 4096);
173   for (i = 0; i < 16; i++)
174     {
175       rt[i] = ng * nb * ((i * 17 * (nr - 1) + 128) >> 8);
176       gt[i] = nb * ((i * 17 * (ng - 1) + 128) >> 8);
177       bt[i] = ((i * 17 * (nb - 1) + 128) >> 8);
178     }
179
180   for (i = 0; i < 4096; i++)
181     {
182       image_info->colorcube[i] = pixels[rt[i >> 8] + gt[(i >> 4) & 0x0f] + bt[i & 0x0f]];
183 #ifdef VERBOSE
184       g_print ("%03x %02x %x %x %x\n", i, image_info->colorcube[i], rt[i >> 8], gt[(i >> 4) & 0x0f], bt[i & 0x0f]);
185 #endif
186     }
187 }
188
189 /* this is the colorcube suitable for dithering */
190 static void
191 gdk_rgb_make_colorcube_d (GdkRgbInfo *image_info, gulong *pixels,
192                           gint nr, gint ng, gint nb)
193 {
194   gint r, g, b;
195   gint i;
196
197   image_info->colorcube_d = g_new (guchar, 512);
198   for (i = 0; i < 512; i++)
199     {
200       r = MIN (nr - 1, i >> 6);
201       g = MIN (ng - 1, (i >> 3) & 7);
202       b = MIN (nb - 1, i & 7);
203       image_info->colorcube_d[i] = pixels[(r * ng + g) * nb + b];
204     }
205 }
206
207 /* Try installing a color cube of the specified size.
208    Make the colorcube and return TRUE on success */
209 static gint
210 gdk_rgb_try_colormap (GdkRgbInfo *image_info, gboolean force,
211                       gint nr, gint ng, gint nb)
212 {
213   gint r, g, b;
214   gint ri, gi, bi;
215   gint r0, g0, b0;
216   GdkColormap *cmap;
217   GdkColor color;
218   gulong pixels[256];
219   gulong junk[256];
220   gint i;
221   gint d2;
222   gint colors_needed;
223   gint idx;
224   gint best[256];
225   GdkScreen *screen;
226
227   if (!force && nr * ng * nb < gdk_rgb_min_colors)
228     return FALSE;
229
230   screen = gdk_visual_get_screen (image_info->visual);
231
232   if (image_info->cmap)
233     cmap = image_info->cmap;
234   else
235     cmap = gdk_screen_get_system_colormap (screen);
236
237   colors_needed = nr * ng * nb;
238   for (i = 0; i < 256; i++)
239     {
240       best[i] = 192;
241       pixels[i] = 256;
242     }
243
244 #ifndef GAMMA
245   if (cmap == gdk_screen_get_system_colormap (screen))
246     /* find color cube colors that are already present */
247     for (i = 0; i < MIN (256, cmap->size); i++)
248       {
249         r = cmap->colors[i].red >> 8;
250         g = cmap->colors[i].green >> 8;
251         b = cmap->colors[i].blue >> 8;
252         ri = (r * (nr - 1) + 128) >> 8;
253         gi = (g * (ng - 1) + 128) >> 8;
254         bi = (b * (nb - 1) + 128) >> 8;
255         r0 = ri * 255 / (nr - 1);
256         g0 = gi * 255 / (ng - 1);
257         b0 = bi * 255 / (nb - 1);
258         idx = ((ri * nr) + gi) * nb + bi;
259         d2 = (r - r0) * (r - r0) + (g - g0) * (g - g0) + (b - b0) * (b - b0);
260         if (d2 < best[idx])
261           {
262             if (pixels[idx] < 256)
263               {
264                 color.pixel = pixels[idx];
265                 gdk_colormap_free_colors (cmap, &color, 1);
266               }
267             else
268               colors_needed--;
269             color = cmap->colors[i];
270             if (!gdk_colormap_alloc_color (cmap, &color, FALSE, FALSE))
271               return gdk_rgb_cmap_fail ("error allocating system color\n",
272                                         cmap, pixels);
273             pixels[idx] = color.pixel; /* which is almost certainly i */
274             best[idx] = d2;
275           }
276       }
277 #endif
278
279   if (colors_needed && 
280       image_info->visual->type != GDK_VISUAL_STATIC_COLOR)
281     {
282       if (!gdk_colors_alloc (cmap, 0, NULL, 0, junk, colors_needed))
283         {
284           char tmp_str[80];
285           
286           sprintf (tmp_str,
287                    "%d %d %d colormap failed (in gdk_colors_alloc)\n",
288                    nr, ng, nb);
289           return gdk_rgb_cmap_fail (tmp_str, cmap, pixels);
290         }
291
292       gdk_colors_free (cmap, junk, colors_needed, 0);
293     }
294
295   for (r = 0, i = 0; r < nr; r++)
296     for (g = 0; g < ng; g++)
297       for (b = 0; b < nb; b++, i++)
298         {
299           if (pixels[i] == 256)
300             {
301               color.red = r * 65535 / (nr - 1);
302               color.green = g * 65535 / (ng - 1);
303               color.blue = b * 65535 / (nb - 1);
304
305 #ifdef GAMMA
306               color.red = 65535 * pow (color.red / 65535.0, 0.5);
307               color.green = 65535 * pow (color.green / 65535.0, 0.5);
308               color.blue = 65535 * pow (color.blue / 65535.0, 0.5);
309 #endif
310
311               if (!gdk_colormap_alloc_color (cmap, &color, FALSE, force))
312                 {
313                   char tmp_str[80];
314
315                   sprintf (tmp_str, "%d %d %d colormap failed\n",
316                            nr, ng, nb);
317                   return gdk_rgb_cmap_fail (tmp_str,
318                                             cmap, pixels);
319                 }
320               pixels[i] = color.pixel;
321             }
322 #ifdef VERBOSE
323           g_print ("%d: %lx\n", i, pixels[i]);
324 #endif
325         }
326
327   image_info->nred_shades = nr;
328   image_info->ngreen_shades = ng;
329   image_info->nblue_shades = nb;
330   gdk_rgb_make_colorcube (image_info, pixels, nr, ng, nb);
331   gdk_rgb_make_colorcube_d (image_info, pixels, nr, ng, nb);
332   return TRUE;
333 }
334
335 /* Return TRUE on success. */
336 static gboolean
337 gdk_rgb_do_colormaps (GdkRgbInfo *image_info, gboolean force)
338 {
339   static const gint sizes[][3] = {
340     /*    { 6, 7, 6 }, */
341     { 6, 6, 6 }, 
342     { 6, 6, 5 }, 
343     { 6, 6, 4 }, 
344     { 5, 5, 5 }, 
345     { 5, 5, 4 }, 
346     { 4, 4, 4 }, 
347     { 4, 4, 3 }, 
348     { 3, 3, 3 }, 
349     { 2, 2, 2 }
350   };
351   static const gint n_sizes = G_N_ELEMENTS (sizes);
352   gint i;
353
354   /* Try the possible sizes. If the force parameter is set to TRUE
355    * and all larger sizes fail, force the larger size to succeed -
356    * this will involve allowing closest matches when allocating the
357    * colors
358    */
359   for (i = 0; i < n_sizes; i++)
360     if (gdk_rgb_try_colormap (image_info,
361                               (i == n_sizes - 1 ) && force,
362                               sizes[i][0], sizes[i][1], sizes[i][2]))
363       return TRUE;
364   return FALSE;
365 }
366
367 /* Make a 2 x 2 x 2 colorcube */
368 static void
369 gdk_rgb_colorcube_222 (GdkRgbInfo *image_info)
370 {
371   int i;
372   GdkColor color;
373
374   image_info->colorcube_d = g_new (guchar, 512);
375
376   for (i = 0; i < 8; i++)
377     {
378       color.red = ((i & 4) >> 2) * 65535;
379       color.green = ((i & 2) >> 1) * 65535;
380       color.blue = (i & 1) * 65535;
381       gdk_colormap_alloc_color (image_info->cmap, &color, FALSE, TRUE);
382       image_info->colorcube_d[((i & 4) << 4) | ((i & 2) << 2) | (i & 1)] = color.pixel;
383     }
384 }
385
386 void
387 gdk_rgb_set_verbose (gboolean verbose)
388 {
389   gdk_rgb_verbose = verbose;
390 }
391
392 void
393 gdk_rgb_set_install (gboolean install)
394 {
395   gdk_rgb_install_cmap = install;
396 }
397
398 void
399 gdk_rgb_set_min_colors (gint min_colors)
400 {
401   gdk_rgb_min_colors = min_colors;
402 }
403
404 /* Return a "score" based on the following criteria (in hex):
405
406    x000 is the quality - 1 is 1bpp, 2 is 4bpp,
407                          4 is 8bpp,
408                          7 is 15bpp truecolor, 8 is 16bpp truecolor,
409                          9 is 24bpp truecolor.
410    0x00 is the speed - 1 is the normal case,
411                        2 means faster than normal
412    00x0 gets a point for being the system visual
413    000x gets a point for being pseudocolor
414
415    A caveat: in the 8bpp modes, being the system visual seems to be
416    quite important. Thus, all of the 8bpp modes should be ranked at
417    the same speed.
418 */
419 static guint32
420 gdk_rgb_score_visual (GdkVisual *visual)
421 {
422   guint32 quality, speed, sys, pseudo;
423
424   quality = 0;
425   speed = 1;
426   sys = 0;
427   if (visual->type == GDK_VISUAL_TRUE_COLOR ||
428       visual->type == GDK_VISUAL_DIRECT_COLOR)
429     {
430       if (visual->depth == 24)
431         {
432           quality = 9;
433           /* Should test for MSB visual here, and set speed if so. */
434         }
435       else if (visual->depth == 16)
436         quality = 8;
437       else if (visual->depth == 15)
438         quality = 7;
439       else if (visual->depth == 8)
440         quality = 4;
441     }
442   else if (visual->type == GDK_VISUAL_PSEUDO_COLOR ||
443            visual->type == GDK_VISUAL_STATIC_COLOR)
444     {
445       if (visual->depth == 8)
446         quality = 4;
447       else if (visual->depth == 4)
448         quality = 2;
449       else if (visual->depth == 1)
450         quality = 1;
451     }
452   else if (visual->type == GDK_VISUAL_STATIC_GRAY
453 #ifdef ENABLE_GRAYSCALE
454            || visual->type == GDK_VISUAL_GRAYSCALE
455 #endif
456            )
457     {
458       if (visual->depth == 8)
459         quality = 4;
460       else if (visual->depth == 4)
461         quality = 2;
462       else if (visual->depth == 1)
463         quality = 1;
464     }
465
466   if (quality == 0)
467     return 0;
468
469   sys = (visual == gdk_screen_get_system_visual (gdk_visual_get_screen (visual)));
470
471   pseudo = (visual->type == GDK_VISUAL_PSEUDO_COLOR || visual->type == GDK_VISUAL_TRUE_COLOR);
472
473   if (gdk_rgb_verbose)
474     g_print ("Visual type = %s, depth = %d, %x:%x:%x%s; score=%x\n",
475              visual_names[visual->type],
476              visual->depth,
477              visual->red_mask,
478              visual->green_mask,
479              visual->blue_mask,
480              sys ? " (system)" : "",
481              (quality << 12) | (speed << 8) | (sys << 4) | pseudo);
482
483   return (quality << 12) | (speed << 8) | (sys << 4) | pseudo;
484 }
485
486 static GdkVisual *
487 gdk_rgb_choose_visual (GdkScreen *screen)
488 {
489   GList *visuals, *tmp_list;
490   guint32 score, best_score;
491   GdkVisual *visual, *best_visual;
492
493   visuals = gdk_screen_list_visuals (screen);
494   tmp_list = visuals;
495
496   best_visual = tmp_list->data;
497   best_score = gdk_rgb_score_visual (best_visual);
498   tmp_list = tmp_list->next;
499   while (tmp_list)
500     {
501       visual = tmp_list->data;
502       score = gdk_rgb_score_visual (visual);
503       if (score > best_score)
504         {
505           best_score = score;
506           best_visual = visual;
507         }
508       tmp_list = tmp_list->next;
509     }
510
511   g_list_free (visuals);
512
513   return best_visual;
514 }
515
516 static void gdk_rgb_select_conv (GdkRgbInfo *image_info);
517
518 static void
519 gdk_rgb_set_gray_cmap (GdkRgbInfo  *image_info,
520                        GdkColormap *cmap)
521 {
522   gint i;
523   GdkColor color;
524   gint status;
525   gulong pixels[256];
526   gint r, g, b, gray;
527
528   for (i = 0; i < 256; i++)
529     {
530       color.pixel = i;
531       color.red = i * 257;
532       color.green = i * 257;
533       color.blue = i * 257;
534       status = gdk_colormap_alloc_color (cmap, &color, FALSE, TRUE);
535       pixels[i] = color.pixel;
536 #ifdef VERBOSE
537       g_print ("allocating pixel %d, %x %x %x, result %d\n",
538                color.pixel, color.red, color.green, color.blue, status);
539 #endif
540     }
541
542   /* Now, we make fake colorcubes - we ultimately just use the pseudocolor
543      methods. */
544
545   image_info->colorcube = g_new (guchar, 4096);
546
547   for (i = 0; i < 4096; i++)
548     {
549       r = (i >> 4) & 0xf0;
550       r = r | r >> 4;
551       g = i & 0xf0;
552       g = g | g >> 4;
553       b = (i << 4 & 0xf0);
554       b = b | b >> 4;
555       gray = (g + ((r + b) >> 1)) >> 1;
556       image_info->colorcube[i] = pixels[gray];
557     }
558 }
559
560 static void
561 gdk_rgb_free_info (GdkRgbInfo *image_info)
562 {
563   GSList *tmp_list;
564   
565   if (image_info->stage_buf)
566     g_free (image_info->stage_buf);
567   
568   if (image_info->gray_cmap)
569     gdk_rgb_cmap_free (image_info->gray_cmap);
570
571   if (image_info->own_gc)
572     g_object_unref (image_info->own_gc);
573
574   if (image_info->colorcube)
575     g_free (image_info->colorcube);
576   
577   if (image_info->colorcube_d)
578     g_free (image_info->colorcube_d);
579
580   tmp_list = image_info->cmap_info_list;
581   while (tmp_list)
582     {
583       GdkRgbCmapInfo *cmap_info = tmp_list->data;
584       cmap_info->cmap->info_list = g_slist_remove (cmap_info->cmap->info_list, cmap_info);
585       g_free (cmap_info);
586     }
587   g_slist_free (image_info->cmap_info_list);
588   
589   g_free (image_info);
590 }
591
592 /* Create a GdkRgbInfo for the given visual/colormap pair. If colormap
593  * is NULL, it will be determined and stored in image_info->cmap. 
594  * In this case, image_info->cmap will have an extra refcount which
595  * is owned by the caller. 
596  */
597 static GdkRgbInfo *
598 gdk_rgb_create_info (GdkVisual *visual, GdkColormap *colormap)
599 {
600   GdkRgbInfo *image_info;
601   GdkScreen *screen = gdk_visual_get_screen (visual);
602
603   image_info = g_new0 (GdkRgbInfo, 1);
604
605   image_info->visual = visual;
606   image_info->cmap = NULL;
607
608   image_info->nred_shades = 6;
609   image_info->ngreen_shades = 6;
610   image_info->nblue_shades = 4;
611   image_info->ngray_shades = 24;
612   image_info->nreserved = 0;
613
614   image_info->bpp = 0;
615   image_info->cmap_alloced = FALSE;
616   image_info->gamma = 1.0;
617
618   image_info->stage_buf = NULL;
619
620   image_info->own_gc = NULL;
621
622   image_info->cmap = colormap;
623
624   /* We used to use the 2x2x2 color cube for pseudo-color with depths
625    * 5, 6, 7 as well but now only use it for depths (3 and) 4 in
626    * pseudo-color. The reason for this is that on Win32 we let the
627    * user restrict the color allocation for PSEUDO_COLOR visuals
628    * (i.e., 256-color mode) and we probably want to do the full
629    * gdk_rgb_do_colormaps() if we are doing that. (Though the color
630    * sharing code won't really be right.)
631    *
632    * (The actual usefulness of this user-requested restriction remains
633    * to be seen, but the code is there in gdkvisual-win32.c. The
634    * thought is that it might occasionally be useful to restrict the
635    * palette size in a GTK application in order to reduce color
636    * flashing.)
637    */
638   if ((image_info->visual->type == GDK_VISUAL_PSEUDO_COLOR &&
639        image_info->visual->depth <= 4 &&
640        image_info->visual->depth >= 3) ||
641       (image_info->visual->type == GDK_VISUAL_STATIC_COLOR &&
642        image_info->visual->depth < 8 &&
643        image_info->visual->depth >= 3))
644     {
645       if (!image_info->cmap)
646         image_info->cmap = g_object_ref (gdk_screen_get_system_colormap (screen));
647       
648       gdk_rgb_colorcube_222 (image_info);
649     }
650   else if (image_info->visual->type == GDK_VISUAL_PSEUDO_COLOR
651     || image_info->visual->type == GDK_VISUAL_STATIC_COLOR)
652     {
653       if (!image_info->cmap &&
654           (gdk_rgb_install_cmap || image_info->visual != gdk_screen_get_system_visual (screen)))
655         {
656           image_info->cmap = gdk_colormap_new (image_info->visual, FALSE);
657           image_info->cmap_alloced = TRUE;
658         }
659       if (!gdk_rgb_do_colormaps (image_info, image_info->cmap != NULL))
660         {
661           image_info->cmap = gdk_colormap_new (image_info->visual, FALSE);
662           image_info->cmap_alloced = TRUE;
663           gdk_rgb_do_colormaps (image_info, TRUE);
664         }
665       if (gdk_rgb_verbose)
666         g_print ("color cube: %d x %d x %d\n",
667                  image_info->nred_shades,
668                  image_info->ngreen_shades,
669                  image_info->nblue_shades);
670
671       if (!image_info->cmap)
672         image_info->cmap = g_object_ref (gdk_screen_get_system_colormap (screen));
673     }
674 #ifdef ENABLE_GRAYSCALE
675   else if (image_info->visual->type == GDK_VISUAL_GRAYSCALE)
676     {
677       if (!image_info->cmap)
678         {
679           image_info->cmap = gdk_colormap_new (image_info->visual, FALSE);
680           image_info->cmap_alloced = TRUE;
681         }
682       
683       gdk_rgb_set_gray_cmap (image_info, image_info->cmap);
684     }
685 #endif
686   else
687     {
688       if (!image_info->cmap)
689         {
690           /* Always install colormap in direct color. */
691           if (image_info->visual->type != GDK_VISUAL_DIRECT_COLOR &&
692               image_info->visual == gdk_screen_get_system_visual (screen))
693             image_info->cmap = g_object_ref (gdk_screen_get_system_colormap (screen));
694           else
695             {
696               image_info->cmap = gdk_colormap_new (image_info->visual, FALSE);
697               image_info->cmap_alloced = TRUE;
698             }
699         }
700     }
701
702   image_info->bitmap = (image_info->visual->depth == 1);
703
704   image_info->bpp = (_gdk_windowing_get_bits_for_depth (gdk_screen_get_display (screen), image_info->visual->depth) + 7) / 8;
705   gdk_rgb_select_conv (image_info);
706
707   if (!gdk_rgb_quark)
708     gdk_rgb_quark = g_quark_from_static_string (gdk_rgb_key);
709
710   g_object_set_qdata_full (G_OBJECT (image_info->cmap), gdk_rgb_quark,
711                            image_info, (GDestroyNotify)gdk_rgb_free_info);
712   return image_info;
713 }
714
715 void
716 gdk_rgb_init (void)
717 {
718   static const gint byte_order[1] = { 1 };
719
720   if (_gdk_debug_flags & GDK_DEBUG_GDKRGB)
721     gdk_rgb_verbose = TRUE;
722
723   /* check endian sanity */
724 #if G_BYTE_ORDER == G_BIG_ENDIAN
725   if (((char *)byte_order)[0] == 1)
726     g_error ("gdk_rgb_init: compiled for big endian, but this is a little endian machine.\n\n");
727 #else
728   if (((char *)byte_order)[0] != 1)
729     g_error ("gdk_rgb_init: compiled for little endian, but this is a big endian machine.\n\n");
730 #endif
731 }
732
733 static GdkRgbInfo *
734 gdk_rgb_get_info_from_colormap (GdkColormap *cmap)
735 {
736   GdkRgbInfo *image_info;
737
738   if (!gdk_rgb_quark)
739     gdk_rgb_quark = g_quark_from_static_string (gdk_rgb_key);
740
741   image_info = g_object_get_qdata (G_OBJECT (cmap), gdk_rgb_quark);
742   if (!image_info)
743     image_info = gdk_rgb_create_info (gdk_colormap_get_visual (cmap), cmap);
744
745   return image_info;
746 }
747
748 static gulong
749 gdk_rgb_xpixel_from_rgb_internal (GdkColormap *colormap,
750                                   guint16 r, guint16 g, guint16 b)
751 {
752   gulong pixel = 0;
753
754   GdkRgbInfo *image_info = gdk_rgb_get_info_from_colormap (colormap);
755
756   if (image_info->bitmap)
757     {
758       return (r + (g << 1) + b) > 131070;
759     }
760   else if (image_info->visual->type == GDK_VISUAL_PSEUDO_COLOR)
761     pixel = image_info->colorcube[((r & 0xf000) >> 4) |
762                                   ((g & 0xf000) >> 8) |
763                                   ((b & 0xf000) >> 12)];
764   else if (image_info->visual->depth < 8 &&
765            image_info->visual->type == GDK_VISUAL_STATIC_COLOR)
766     {
767       pixel = image_info->colorcube_d[((r & 0x8000) >> 9) |
768                                       ((g & 0x8000) >> 12) |
769                                       ((b & 0x8000) >> 15)];
770     }
771   else if (image_info->visual->type == GDK_VISUAL_TRUE_COLOR ||
772            image_info->visual->type == GDK_VISUAL_DIRECT_COLOR)
773     {
774 #ifdef VERBOSE
775       g_print ("shift, prec: r %d %d g %d %d b %d %d\n",
776                image_info->visual->red_shift,
777                image_info->visual->red_prec,
778                image_info->visual->green_shift,
779                image_info->visual->green_prec,
780                image_info->visual->blue_shift,
781                image_info->visual->blue_prec);
782 #endif
783
784       pixel = (((r >> (16 - image_info->visual->red_prec)) << image_info->visual->red_shift) +
785                ((g >> (16 - image_info->visual->green_prec)) << image_info->visual->green_shift) +
786                ((b >> (16 - image_info->visual->blue_prec)) << image_info->visual->blue_shift));
787     }
788   else if (image_info->visual->type == GDK_VISUAL_STATIC_GRAY ||
789            image_info->visual->type == GDK_VISUAL_GRAYSCALE)
790     {
791       int gray = r + g * 2 + b;
792       return gray >> (18 - image_info->visual->depth);
793     }
794
795   return pixel;
796 }
797
798 /* convert an rgb value into an X pixel code */
799 gulong
800 gdk_rgb_xpixel_from_rgb (guint32 rgb)
801 {
802   guint32 r = rgb & 0xff0000;
803   guint32 g = rgb & 0xff00;
804   guint32 b = rgb & 0xff;
805
806   return gdk_rgb_xpixel_from_rgb_internal (gdk_screen_get_rgb_colormap (gdk_screen_get_default ()),
807                                            (r >> 8) + (r >> 16), g + (g >> 8), b + (b << 8));
808 }
809
810 void
811 gdk_rgb_gc_set_foreground (GdkGC *gc, guint32 rgb)
812 {
813   GdkColor color;
814
815   color.pixel = gdk_rgb_xpixel_from_rgb (rgb);
816   gdk_gc_set_foreground (gc, &color);
817 }
818
819 void
820 gdk_rgb_gc_set_background (GdkGC *gc, guint32 rgb)
821 {
822   GdkColor color;
823
824   color.pixel = gdk_rgb_xpixel_from_rgb (rgb);
825   gdk_gc_set_background (gc, &color);
826 }
827
828 /**
829  * gdk_rgb_find_color:
830  * @colormap: a #GdkColormap
831  * @color: a #GdkColor
832  *
833  * @colormap should be the colormap for the graphics context and
834  * drawable you're using to draw. If you're drawing to a #GtkWidget,
835  * call gtk_widget_get_colormap().
836  *
837  * @color should have its %red, %green, and %blue fields initialized;
838  * gdk_rgb_find_color() will fill in the %pixel field with the best
839  * matching pixel from a color cube. The color is then ready to be
840  * used for drawing, e.g. you can call gdk_gc_set_foreground() which
841  * expects %pixel to be initialized.
842  *
843  * In many cases, you can avoid this whole issue by calling
844  * gdk_gc_set_rgb_fg_color() or gdk_gc_set_rgb_bg_color(), which
845  * do not expect %pixel to be initialized in advance. If you use those
846  * functions, there's no need for gdk_rgb_find_color().
847  * 
848  **/
849 void
850 gdk_rgb_find_color (GdkColormap *colormap, GdkColor *color)
851 {
852   color->pixel = gdk_rgb_xpixel_from_rgb_internal (colormap,
853                                                    color->red, color->green, color->blue);
854 }
855
856 #if G_BYTE_ORDER == G_LITTLE_ENDIAN
857 #define HAIRY_CONVERT_8
858 #endif
859
860 #ifdef HAIRY_CONVERT_8
861 static void
862 gdk_rgb_convert_8 (GdkRgbInfo *image_info, GdkImage *image,
863                    gint x0, gint y0, gint width, gint height,
864                    guchar *buf, int rowstride,
865                    gint x_align, gint y_align, GdkRgbCmap *cmap)
866 {
867   int x, y;
868   gint bpl;
869   guchar *obuf, *obptr;
870   guchar *bptr, *bp2;
871   gint r, g, b;
872   guchar *colorcube = image_info->colorcube;
873
874   bptr = buf;
875   bpl = image->bpl;
876   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
877   for (y = 0; y < height; y++)
878     {
879       bp2 = bptr;
880       obptr = obuf;
881       if (((unsigned long)obuf | (unsigned long) bp2) & 3)
882         {
883           for (x = 0; x < width; x++)
884             {
885               r = *bp2++;
886               g = *bp2++;
887               b = *bp2++;
888               obptr[0] = colorcube[((r & 0xf0) << 4) |
889                                   (g & 0xf0) |
890                                   (b >> 4)];
891               obptr++;
892             }
893         }
894       else
895         {
896           for (x = 0; x < width - 3; x += 4)
897             {
898               guint32 r1b0g0r0;
899               guint32 g2r2b1g1;
900               guint32 b3g3r3b2;
901
902               r1b0g0r0 = ((guint32 *)bp2)[0];
903               g2r2b1g1 = ((guint32 *)bp2)[1];
904               b3g3r3b2 = ((guint32 *)bp2)[2];
905               ((guint32 *)obptr)[0] =
906                 colorcube[((r1b0g0r0 & 0xf0) << 4) | 
907                          ((r1b0g0r0 & 0xf000) >> 8) |
908                          ((r1b0g0r0 & 0xf00000) >> 20)] |
909                 (colorcube[((r1b0g0r0 & 0xf0000000) >> 20) |
910                           (g2r2b1g1 & 0xf0) |
911                           ((g2r2b1g1 & 0xf000) >> 12)] << 8) |
912                 (colorcube[((g2r2b1g1 & 0xf00000) >> 12) |
913                           ((g2r2b1g1 & 0xf0000000) >> 24) |
914                           ((b3g3r3b2 & 0xf0) >> 4)] << 16) |
915                 (colorcube[((b3g3r3b2 & 0xf000) >> 4) |
916                           ((b3g3r3b2 & 0xf00000) >> 16) |
917                           (b3g3r3b2 >> 28)] << 24);
918               bp2 += 12;
919               obptr += 4;
920             }
921           for (; x < width; x++)
922             {
923               r = *bp2++;
924               g = *bp2++;
925               b = *bp2++;
926               obptr[0] = colorcube[((r & 0xf0) << 4) |
927                                   (g & 0xf0) |
928                                   (b >> 4)];
929               obptr++;
930             }
931         }
932       bptr += rowstride;
933       obuf += bpl;
934     }
935 }
936 #else
937 static void
938 gdk_rgb_convert_8 (GdkRgbInfo *image_info, GdkImage *image,
939                    gint x0, gint y0, gint width, gint height,
940                    guchar *buf, int rowstride,
941                    gint x_align, gint y_align, GdkRgbCmap *cmap)
942 {
943   int x, y;
944   gint bpl;
945   guchar *obuf, *obptr;
946   guchar *bptr, *bp2;
947   gint r, g, b;
948   guchar *colorcube = image_info->colorcube;
949
950   bptr = buf;
951   bpl = image->bpl;
952   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
953   for (y = 0; y < height; y++)
954     {
955       bp2 = bptr;
956       obptr = obuf;
957       for (x = 0; x < width; x++)
958         {
959           r = *bp2++;
960           g = *bp2++;
961           b = *bp2++;
962           obptr[0] = colorcube[((r & 0xf0) << 4) |
963                               (g & 0xf0) |
964                               (b >> 4)];
965           obptr++;
966         }
967       bptr += rowstride;
968       obuf += bpl;
969     }
970 }
971 #endif
972
973 #if 1
974
975 /* This dither table was generated by Raph Levien using patented
976    technology (US Patent 5,276,535). The dither table itself is in the
977    public domain. */
978
979 #define DM_WIDTH 128
980 #define DM_WIDTH_SHIFT 7
981 #define DM_HEIGHT 128
982 static const guchar DM[128][128] =
983 {
984   { 0, 41, 23, 5, 17, 39, 7, 15, 62, 23, 40, 51, 31, 47, 9, 32, 52, 27, 57, 25, 6, 61, 27, 52, 37, 7, 40, 63, 18, 36, 10, 42, 25, 62, 45, 34, 20, 42, 37, 14, 35, 29, 50, 10, 61, 2, 40, 8, 37, 12, 58, 22, 5, 41, 10, 39, 0, 60, 11, 46, 2, 55, 38, 17, 36, 59, 13, 54, 37, 56, 8, 29, 16, 13, 63, 22, 41, 55, 7, 20, 49, 14, 23, 55, 37, 23, 19, 36, 15, 49, 23, 63, 30, 14, 38, 27, 53, 13, 22, 41, 19, 31, 7, 19, 50, 30, 49, 16, 3, 32, 56, 40, 29, 34, 8, 48, 19, 45, 4, 51, 12, 46, 35, 49, 16, 42, 12, 62 },
985   { 30, 57, 36, 54, 47, 34, 52, 27, 43, 4, 28, 7, 17, 36, 62, 13, 44, 7, 18, 48, 33, 21, 44, 14, 30, 47, 12, 33, 5, 55, 31, 58, 13, 30, 4, 17, 52, 10, 60, 26, 46, 0, 39, 27, 42, 22, 47, 25, 60, 32, 9, 38, 48, 17, 59, 30, 49, 18, 34, 25, 51, 19, 5, 48, 21, 8, 28, 46, 1, 32, 41, 19, 54, 47, 37, 18, 28, 11, 44, 30, 39, 56, 2, 33, 8, 42, 61, 28, 58, 8, 46, 9, 41, 4, 58, 7, 21, 48, 59, 10, 52, 14, 42, 57, 12, 25, 7, 53, 42, 24, 11, 50, 17, 59, 42, 2, 36, 60, 32, 17, 63, 29, 21, 7, 59, 32, 24, 39 },
986   { 22, 8, 16, 32, 3, 25, 13, 57, 18, 45, 58, 39, 55, 20, 5, 42, 23, 34, 63, 1, 51, 10, 58, 4, 60, 23, 53, 27, 44, 21, 3, 48, 8, 50, 43, 54, 27, 32, 5, 55, 21, 58, 12, 53, 6, 36, 14, 50, 17, 29, 53, 15, 24, 52, 7, 36, 13, 42, 4, 53, 9, 35, 61, 26, 56, 32, 49, 15, 62, 23, 6, 60, 2, 31, 4, 48, 58, 38, 15, 61, 5, 25, 47, 28, 50, 15, 7, 40, 3, 32, 33, 52, 25, 50, 35, 42, 61, 3, 28, 36, 23, 63, 4, 33, 46, 62, 36, 23, 60, 6, 54, 28, 4, 37, 23, 55, 25, 8, 42, 54, 14, 6, 56, 38, 19, 52, 4, 46 },
987   { 48, 53, 43, 12, 45, 63, 30, 37, 9, 34, 21, 1, 25, 47, 29, 58, 3, 54, 15, 39, 29, 17, 38, 35, 20, 43, 1, 49, 15, 59, 29, 39, 22, 35, 16, 23, 1, 47, 39, 18, 8, 44, 25, 31, 57, 19, 63, 4, 45, 3, 42, 61, 1, 31, 45, 20, 57, 29, 62, 21, 32, 41, 14, 44, 3, 39, 5, 34, 10, 43, 51, 35, 23, 52, 40, 10, 21, 1, 53, 18, 51, 43, 12, 62, 18, 54, 26, 51, 20, 57, 14, 1, 62, 16, 11, 18, 32, 39, 17, 44, 1, 48, 26, 37, 18, 2, 51, 14, 28, 45, 35, 18, 57, 13, 47, 11, 51, 20, 2, 39, 31, 47, 25, 1, 50, 11, 60, 7 },
988   { 18, 28, 1, 56, 21, 10, 51, 2, 46, 54, 14, 61, 11, 50, 13, 38, 19, 31, 45, 9, 55, 24, 47, 5, 54, 9, 62, 11, 35, 8, 51, 14, 57, 6, 63, 40, 58, 14, 51, 28, 62, 34, 15, 48, 1, 41, 30, 35, 55, 21, 34, 11, 49, 37, 8, 52, 4, 23, 15, 43, 1, 58, 11, 23, 53, 16, 55, 26, 58, 18, 27, 12, 45, 14, 25, 63, 42, 33, 27, 35, 9, 31, 21, 38, 1, 44, 34, 12, 48, 38, 21, 44, 29, 47, 26, 53, 1, 46, 54, 8, 59, 29, 11, 55, 22, 41, 33, 20, 39, 1, 48, 9, 44, 32, 5, 62, 29, 44, 57, 23, 10, 58, 34, 43, 15, 37, 26, 33 },
989   { 51, 38, 59, 24, 35, 42, 19, 60, 5, 32, 41, 26, 43, 33, 7, 53, 48, 11, 59, 23, 42, 2, 61, 30, 16, 40, 32, 24, 56, 41, 19, 33, 37, 26, 47, 9, 31, 22, 2, 45, 9, 54, 4, 37, 21, 52, 11, 23, 7, 57, 16, 25, 55, 18, 63, 27, 46, 39, 56, 10, 50, 37, 29, 47, 19, 63, 24, 9, 46, 2, 39, 60, 9, 57, 30, 7, 49, 11, 59, 3, 45, 57, 5, 60, 29, 22, 5, 60, 30, 9, 59, 18, 40, 6, 57, 36, 30, 12, 24, 34, 15, 40, 52, 6, 49, 9, 58, 4, 63, 12, 26, 61, 22, 53, 38, 16, 35, 14, 28, 50, 42, 17, 5, 28, 62, 20, 54, 12 },
990   { 26, 6, 31, 15, 49, 6, 38, 27, 22, 49, 16, 56, 2, 62, 30, 21, 0, 36, 28, 6, 49, 32, 13, 52, 26, 50, 19, 46, 3, 26, 62, 0, 53, 12, 29, 3, 53, 41, 60, 24, 38, 13, 58, 16, 43, 9, 59, 39, 46, 28, 44, 40, 2, 33, 13, 41, 16, 6, 47, 31, 26, 17, 57, 6, 38, 0, 42, 36, 29, 52, 20, 31, 48, 0, 34, 56, 20, 36, 23, 54, 14, 41, 24, 37, 10, 55, 46, 25, 16, 45, 36, 4, 55, 23, 15, 8, 50, 62, 5, 56, 44, 20, 13, 28, 59, 31, 24, 47, 31, 52, 37, 17, 40, 0, 26, 49, 3, 60, 7, 33, 0, 61, 53, 40, 8, 45, 2, 41 },
991   { 16, 63, 43, 4, 61, 24, 56, 13, 53, 8, 36, 12, 24, 41, 16, 46, 60, 26, 52, 39, 14, 57, 21, 37, 0, 45, 7, 59, 38, 17, 43, 10, 45, 20, 61, 43, 19, 11, 33, 17, 50, 32, 23, 61, 28, 49, 26, 0, 18, 51, 5, 60, 22, 58, 29, 0, 59, 34, 19, 62, 3, 52, 7, 44, 30, 59, 13, 50, 15, 62, 7, 17, 38, 22, 44, 15, 40, 4, 47, 28, 33, 17, 49, 16, 51, 40, 10, 56, 0, 53, 13, 49, 28, 38, 60, 21, 43, 19, 37, 27, 3, 51, 34, 39, 0, 45, 15, 43, 10, 21, 3, 55, 8, 33, 59, 10, 41, 18, 52, 24, 46, 20, 30, 13, 58, 22, 36, 57 },
992   { 50, 34, 11, 47, 29, 17, 44, 0, 33, 63, 28, 46, 52, 5, 57, 10, 42, 18, 4, 63, 20, 8, 44, 10, 56, 34, 14, 29, 5, 54, 23, 59, 32, 49, 7, 34, 49, 27, 56, 0, 42, 7, 46, 3, 40, 6, 54, 32, 62, 13, 36, 10, 47, 8, 35, 49, 24, 51, 12, 40, 22, 35, 60, 12, 22, 51, 33, 4, 40, 25, 43, 55, 5, 54, 12, 61, 26, 51, 8, 62, 0, 53, 7, 63, 2, 32, 19, 34, 42, 24, 31, 63, 2, 10, 45, 33, 0, 48, 9, 61, 22, 47, 8, 62, 18, 56, 7, 54, 27, 57, 46, 30, 50, 19, 45, 30, 56, 36, 22, 47, 11, 38, 3, 51, 32, 48, 18, 9 },
993   { 0, 21, 40, 19, 52, 9, 37, 48, 20, 40, 3, 18, 27, 38, 35, 22, 31, 56, 13, 35, 46, 28, 60, 40, 27, 18, 61, 50, 41, 30, 7, 36, 2, 25, 16, 57, 5, 15, 47, 29, 55, 19, 30, 52, 15, 34, 20, 12, 43, 30, 20, 54, 25, 44, 53, 12, 38, 5, 55, 27, 48, 15, 33, 27, 45, 8, 19, 28, 56, 11, 33, 49, 18, 36, 29, 2, 45, 16, 39, 19, 31, 43, 27, 35, 20, 52, 26, 6, 61, 11, 41, 17, 29, 51, 20, 56, 25, 32, 41, 17, 53, 31, 25, 14, 42, 23, 35, 16, 38, 6, 34, 12, 15, 62, 6, 21, 13, 1, 63, 9, 55, 27, 43, 25, 14, 4, 31, 55 },
994   { 44, 29, 61, 2, 35, 58, 26, 15, 60, 10, 51, 59, 14, 55, 8, 50, 2, 44, 25, 51, 1, 33, 16, 4, 48, 36, 2, 21, 12, 57, 48, 13, 51, 55, 40, 28, 37, 62, 8, 39, 12, 63, 36, 10, 59, 24, 56, 47, 9, 50, 41, 1, 32, 17, 6, 21, 61, 30, 9, 43, 1, 54, 41, 2, 54, 37, 48, 61, 1, 46, 21, 3, 58, 24, 50, 32, 60, 10, 57, 25, 46, 12, 59, 4, 45, 13, 57, 47, 27, 39, 5, 58, 47, 14, 35, 4, 52, 13, 60, 6, 36, 10, 45, 55, 4, 50, 29, 2, 61, 50, 25, 58, 44, 24, 36, 42, 54, 28, 40, 32, 16, 56, 6, 62, 46, 39, 60, 23 },
995   { 7, 48, 14, 54, 23, 40, 4, 45, 30, 22, 42, 32, 1, 44, 20, 29, 58, 8, 37, 19, 41, 54, 24, 58, 9, 53, 25, 46, 34, 16, 23, 38, 27, 11, 18, 1, 52, 21, 35, 22, 48, 5, 25, 45, 18, 38, 2, 27, 35, 4, 57, 15, 62, 39, 57, 28, 42, 16, 36, 60, 24, 18, 10, 63, 20, 5, 16, 23, 37, 14, 59, 27, 41, 8, 13, 42, 21, 35, 6, 50, 3, 38, 15, 48, 30, 39, 17, 3, 49, 14, 53, 33, 24, 7, 61, 44, 11, 39, 23, 49, 19, 58, 1, 32, 36, 12, 60, 41, 20, 13, 41, 4, 39, 1, 48, 8, 18, 51, 14, 44, 5, 37, 21, 34, 1, 26, 10, 37 },
996   { 53, 36, 27, 9, 50, 12, 32, 55, 2, 57, 7, 17, 48, 34, 63, 15, 40, 26, 62, 11, 49, 6, 31, 39, 22, 42, 6, 63, 1, 39, 60, 4, 42, 61, 32, 45, 24, 44, 2, 60, 16, 41, 53, 1, 33, 61, 49, 17, 63, 23, 45, 26, 33, 3, 23, 46, 2, 50, 20, 4, 45, 34, 49, 30, 39, 58, 44, 31, 53, 34, 6, 52, 30, 47, 63, 1, 53, 22, 42, 31, 58, 23, 54, 22, 61, 8, 36, 59, 22, 35, 21, 1, 55, 40, 27, 16, 30, 54, 2, 29, 43, 16, 39, 63, 21, 46, 26, 10, 48, 32, 19, 53, 30, 56, 26, 60, 33, 4, 61, 23, 49, 59, 15, 53, 19, 58, 42, 16 },
997   { 20, 5, 59, 46, 25, 62, 7, 19, 43, 25, 37, 61, 11, 24, 4, 54, 12, 52, 3, 32, 17, 61, 12, 47, 15, 55, 18, 31, 53, 28, 9, 50, 21, 6, 55, 9, 58, 14, 54, 26, 33, 7, 31, 58, 13, 21, 8, 42, 29, 6, 37, 11, 48, 52, 14, 60, 11, 39, 56, 32, 14, 58, 7, 26, 17, 4, 42, 8, 11, 47, 19, 38, 10, 17, 26, 37, 9, 55, 28, 13, 18, 40, 6, 33, 1, 43, 25, 11, 51, 7, 62, 43, 18, 37, 3, 57, 45, 9, 38, 58, 5, 52, 27, 7, 17, 53, 5, 57, 37, 2, 63, 9, 22, 15, 11, 38, 25, 45, 35, 0, 28, 10, 41, 30, 50, 8, 31, 57 },
998   { 49, 33, 16, 38, 1, 42, 51, 34, 53, 14, 28, 49, 30, 56, 36, 23, 43, 20, 38, 56, 22, 45, 28, 0, 62, 35, 26, 44, 11, 19, 52, 35, 44, 15, 30, 38, 10, 31, 40, 4, 46, 50, 20, 40, 27, 44, 51, 14, 56, 53, 19, 59, 7, 29, 41, 19, 35, 25, 8, 52, 22, 44, 13, 53, 50, 32, 61, 24, 56, 25, 63, 0, 45, 57, 33, 59, 16, 46, 4, 62, 50, 11, 60, 37, 52, 19, 55, 29, 37, 46, 13, 26, 48, 10, 50, 34, 21, 63, 26, 13, 42, 33, 22, 55, 35, 28, 43, 15, 24, 51, 27, 34, 46, 49, 58, 3, 52, 9, 57, 19, 48, 55, 3, 35, 12, 45, 24, 3 },
999   { 41, 11, 56, 28, 18, 31, 22, 10, 37, 6, 47, 13, 3, 41, 9, 46, 0, 48, 29, 6, 34, 10, 55, 37, 20, 8, 49, 3, 41, 59, 14, 25, 0, 63, 19, 47, 27, 51, 17, 57, 23, 10, 61, 6, 54, 3, 38, 31, 0, 22, 34, 43, 20, 55, 31, 0, 49, 63, 29, 38, 3, 62, 28, 40, 0, 22, 14, 35, 2, 48, 15, 43, 23, 14, 3, 29, 49, 20, 39, 34, 0, 44, 29, 9, 15, 47, 5, 42, 0, 31, 58, 5, 31, 61, 23, 15, 0, 47, 19, 50, 24, 3, 59, 11, 44, 0, 31, 59, 6, 42, 17, 60, 0, 39, 20, 31, 43, 17, 29, 40, 12, 25, 60, 22, 52, 15, 63, 29 },
1000   { 20, 52, 8, 44, 62, 4, 59, 49, 17, 63, 21, 39, 60, 18, 52, 27, 33, 59, 14, 51, 59, 43, 24, 5, 51, 30, 57, 17, 32, 5, 37, 56, 48, 34, 42, 3, 60, 5, 36, 13, 43, 37, 18, 34, 25, 12, 59, 24, 47, 36, 11, 50, 3, 38, 9, 58, 16, 5, 43, 18, 47, 10, 37, 18, 59, 46, 29, 52, 40, 12, 34, 28, 56, 36, 53, 7, 43, 8, 24, 52, 26, 17, 56, 43, 24, 32, 63, 20, 57, 16, 22, 52, 36, 8, 41, 56, 29, 32, 54, 7, 35, 57, 14, 48, 20, 62, 13, 39, 53, 29, 8, 45, 13, 29, 7, 61, 14, 54, 6, 63, 38, 32, 18, 43, 2, 39, 6, 47 },
1001   { 0, 58, 23, 35, 13, 46, 12, 39, 0, 31, 55, 24, 5, 35, 15, 61, 17, 5, 39, 25, 18, 2, 50, 33, 41, 13, 39, 23, 62, 46, 29, 12, 22, 8, 56, 25, 20, 49, 32, 62, 0, 56, 11, 46, 63, 42, 9, 16, 55, 5, 60, 15, 62, 26, 45, 21, 36, 51, 13, 57, 31, 24, 55, 6, 35, 9, 57, 5, 20, 60, 7, 51, 5, 19, 40, 25, 61, 32, 56, 12, 36, 48, 21, 2, 58, 12, 39, 28, 9, 50, 40, 12, 44, 18, 25, 49, 6, 38, 11, 62, 18, 46, 30, 9, 40, 25, 49, 19, 10, 36, 55, 22, 33, 52, 41, 18, 37, 27, 49, 21, 2, 46, 7, 53, 33, 61, 27, 35 },
1002   { 41, 31, 5, 39, 51, 26, 33, 57, 27, 41, 9, 44, 54, 29, 48, 7, 44, 36, 57, 10, 31, 63, 16, 45, 11, 60, 1, 47, 7, 20, 43, 3, 58, 36, 13, 52, 39, 7, 15, 28, 22, 48, 30, 21, 1, 29, 49, 44, 27, 17, 40, 30, 24, 42, 12, 53, 33, 7, 47, 20, 1, 42, 11, 49, 25, 43, 17, 32, 45, 27, 41, 21, 31, 62, 11, 49, 2, 15, 42, 5, 63, 7, 41, 27, 49, 6, 54, 23, 46, 34, 2, 28, 54, 3, 59, 12, 46, 17, 42, 28, 40, 1, 37, 51, 5, 55, 2, 34, 47, 16, 3, 62, 47, 5, 23, 56, 1, 44, 12, 34, 51, 16, 57, 11, 25, 17, 54, 13 },
1003   { 60, 26, 55, 18, 3, 60, 20, 6, 52, 15, 50, 19, 32, 11, 23, 53, 26, 21, 1, 47, 42, 27, 8, 58, 21, 27, 53, 36, 26, 54, 31, 50, 17, 30, 45, 1, 29, 59, 44, 53, 41, 4, 35, 58, 51, 19, 32, 4, 52, 34, 48, 8, 51, 5, 56, 2, 25, 61, 27, 38, 54, 27, 62, 21, 51, 1, 39, 62, 10, 50, 1, 58, 13, 47, 38, 18, 35, 54, 22, 51, 30, 19, 59, 34, 14, 32, 44, 4, 60, 15, 52, 62, 20, 43, 30, 35, 21, 60, 4, 52, 12, 24, 61, 18, 30, 42, 23, 61, 25, 50, 27, 38, 11, 59, 12, 35, 50, 30, 59, 24, 8, 42, 28, 37, 48, 9, 44, 21 },
1004   { 10, 47, 15, 50, 30, 43, 8, 45, 29, 2, 36, 59, 1, 58, 41, 3, 63, 31, 54, 20, 13, 55, 35, 38, 4, 44, 15, 9, 61, 2, 14, 38, 61, 10, 23, 54, 18, 12, 24, 2, 14, 55, 16, 8, 38, 14, 41, 60, 10, 23, 1, 58, 32, 17, 28, 37, 41, 15, 3, 60, 15, 33, 4, 36, 16, 59, 28, 14, 23, 55, 37, 18, 44, 28, 2, 57, 30, 10, 27, 46, 14, 38, 3, 53, 21, 61, 17, 35, 10, 41, 26, 7, 33, 9, 57, 1, 53, 37, 26, 20, 56, 48, 9, 33, 58, 16, 37, 7, 45, 1, 57, 15, 32, 26, 42, 23, 7, 20, 4, 54, 31, 62, 22, 1, 59, 30, 4, 51 },
1005   { 36, 2, 38, 11, 24, 36, 54, 22, 62, 47, 25, 8, 28, 45, 16, 38, 12, 43, 9, 37, 49, 3, 23, 52, 18, 30, 50, 33, 19, 42, 49, 26, 6, 40, 47, 35, 63, 38, 50, 33, 60, 26, 36, 47, 24, 57, 6, 26, 39, 63, 19, 44, 14, 46, 61, 9, 50, 30, 45, 23, 10, 50, 44, 8, 31, 54, 6, 46, 36, 4, 30, 54, 8, 52, 22, 41, 4, 60, 40, 0, 58, 24, 45, 10, 37, 1, 48, 30, 56, 17, 38, 48, 24, 47, 19, 39, 14, 8, 45, 32, 2, 34, 27, 44, 4, 52, 11, 56, 31, 21, 40, 19, 44, 51, 2, 63, 46, 58, 36, 43, 14, 5, 50, 38, 14, 56, 40, 23 },
1006   { 61, 46, 32, 63, 54, 1, 14, 34, 12, 40, 18, 49, 37, 10, 61, 30, 51, 24, 60, 7, 29, 40, 62, 11, 46, 58, 6, 56, 24, 10, 34, 52, 21, 59, 16, 3, 27, 5, 20, 46, 9, 40, 7, 62, 2, 30, 53, 15, 48, 10, 28, 35, 54, 6, 21, 34, 18, 55, 7, 40, 57, 19, 26, 60, 41, 13, 24, 51, 19, 61, 9, 25, 34, 15, 63, 11, 45, 17, 20, 47, 33, 8, 31, 62, 43, 26, 53, 7, 24, 59, 0, 13, 55, 4, 62, 27, 51, 31, 63, 15, 58, 7, 54, 14, 46, 22, 28, 43, 12, 63, 8, 54, 5, 17, 39, 33, 15, 10, 27, 17, 47, 34, 19, 45, 27, 12, 33, 17 },
1007   { 5, 28, 21, 7, 17, 48, 42, 58, 23, 4, 63, 14, 55, 21, 34, 5, 19, 0, 45, 17, 52, 15, 25, 32, 0, 22, 40, 13, 45, 62, 18, 0, 43, 11, 33, 55, 30, 42, 57, 19, 51, 31, 22, 43, 18, 45, 34, 0, 43, 31, 56, 3, 23, 40, 59, 0, 44, 13, 48, 35, 2, 32, 46, 0, 21, 48, 35, 3, 40, 32, 43, 59, 0, 48, 33, 26, 53, 36, 55, 12, 51, 16, 55, 5, 18, 29, 11, 39, 51, 19, 45, 31, 42, 21, 35, 6, 22, 47, 10, 38, 23, 50, 20, 36, 0, 60, 38, 4, 50, 35, 48, 34, 24, 57, 9, 53, 28, 48, 61, 0, 56, 24, 53, 3, 63, 6, 42, 57 },
1008   { 13, 53, 45, 40, 58, 27, 6, 16, 38, 51, 33, 30, 43, 2, 47, 56, 40, 50, 33, 57, 27, 5, 47, 42, 60, 36, 16, 54, 28, 4, 37, 57, 28, 51, 22, 8, 45, 14, 6, 39, 0, 54, 11, 59, 28, 12, 50, 21, 61, 13, 19, 38, 49, 11, 25, 37, 58, 29, 22, 63, 14, 56, 12, 53, 30, 63, 9, 57, 26, 12, 47, 16, 23, 39, 50, 6, 31, 2, 25, 6, 28, 41, 36, 22, 50, 57, 42, 3, 34, 8, 28, 61, 11, 50, 16, 54, 41, 0, 55, 43, 5, 29, 41, 63, 25, 16, 53, 18, 26, 10, 21, 0, 61, 30, 41, 22, 3, 38, 20, 39, 29, 8, 41, 16, 36, 52, 22, 19 },
1009   { 55, 34, 0, 25, 10, 32, 56, 44, 28, 0, 57, 7, 26, 53, 23, 8, 13, 35, 22, 12, 36, 60, 20, 8, 14, 29, 48, 2, 41, 49, 23, 13, 39, 7, 48, 58, 25, 53, 34, 62, 28, 16, 48, 4, 37, 56, 27, 5, 36, 52, 46, 7, 62, 33, 52, 11, 17, 53, 5, 28, 41, 24, 38, 17, 5, 39, 20, 45, 15, 56, 5, 38, 60, 8, 14, 57, 21, 48, 62, 39, 59, 13, 1, 60, 9, 32, 16, 63, 44, 25, 52, 15, 36, 2, 60, 29, 12, 33, 25, 17, 59, 45, 13, 8, 49, 32, 6, 40, 59, 29, 45, 37, 13, 47, 6, 55, 30, 45, 9, 52, 13, 59, 25, 47, 32, 1, 49, 30 },
1010   { 9, 39, 14, 61, 49, 37, 3, 20, 50, 13, 41, 19, 46, 17, 38, 59, 28, 62, 4, 44, 54, 1, 34, 51, 55, 7, 63, 32, 21, 8, 56, 31, 62, 19, 36, 1, 41, 17, 24, 12, 42, 35, 25, 52, 20, 8, 44, 59, 25, 2, 22, 42, 16, 29, 4, 46, 20, 36, 43, 9, 51, 8, 49, 26, 58, 33, 54, 1, 37, 29, 52, 20, 27, 45, 19, 35, 42, 16, 10, 32, 20, 49, 46, 27, 40, 4, 47, 22, 13, 55, 4, 47, 26, 44, 23, 40, 58, 19, 48, 13, 31, 2, 57, 34, 42, 19, 61, 32, 14, 55, 5, 51, 26, 19, 58, 16, 49, 14, 62, 5, 33, 44, 21, 7, 60, 26, 11, 41 },
1011   { 62, 24, 47, 29, 8, 19, 53, 11, 60, 24, 32, 61, 4, 55, 31, 2, 49, 16, 39, 9, 31, 24, 43, 17, 26, 38, 11, 25, 58, 43, 12, 35, 3, 46, 15, 32, 63, 4, 49, 56, 2, 60, 10, 32, 63, 17, 39, 12, 55, 30, 57, 9, 48, 55, 39, 24, 60, 2, 58, 31, 19, 61, 34, 3, 42, 11, 22, 46, 7, 61, 10, 42, 3, 55, 32, 1, 58, 28, 44, 54, 4, 34, 23, 15, 56, 20, 37, 58, 6, 30, 38, 18, 63, 9, 32, 5, 51, 3, 62, 37, 52, 18, 39, 23, 3, 51, 9, 47, 1, 23, 43, 15, 60, 35, 11, 40, 1, 36, 31, 26, 57, 2, 37, 54, 18, 44, 58, 16 },
1012   { 5, 51, 3, 33, 43, 62, 21, 42, 35, 9, 48, 15, 36, 10, 22, 42, 20, 46, 26, 56, 50, 12, 59, 3, 48, 19, 45, 53, 1, 27, 47, 17, 52, 24, 56, 11, 51, 21, 37, 30, 20, 46, 14, 41, 1, 47, 33, 7, 41, 17, 35, 27, 20, 1, 14, 54, 26, 33, 18, 47, 1, 44, 14, 59, 16, 52, 28, 18, 49, 31, 25, 34, 63, 13, 51, 24, 9, 50, 3, 23, 38, 63, 7, 52, 29, 46, 11, 33, 50, 22, 57, 36, 1, 57, 49, 17, 39, 28, 9, 35, 6, 27, 53, 15, 55, 30, 24, 58, 36, 41, 11, 52, 32, 3, 44, 25, 62, 23, 51, 15, 42, 22, 50, 10, 39, 4, 31, 35 },
1013   { 46, 22, 57, 17, 12, 39, 26, 5, 31, 59, 1, 45, 27, 62, 52, 7, 58, 33, 6, 18, 39, 22, 33, 41, 57, 5, 35, 18, 40, 16, 60, 5, 29, 42, 7, 39, 27, 44, 9, 47, 8, 26, 54, 22, 51, 29, 24, 49, 15, 61, 4, 51, 31, 63, 43, 6, 50, 8, 39, 12, 53, 37, 23, 30, 40, 6, 62, 43, 14, 53, 2, 49, 7, 36, 17, 41, 61, 37, 18, 56, 11, 18, 44, 35, 2, 19, 61, 0, 41, 14, 8, 30, 43, 12, 24, 46, 14, 54, 42, 21, 44, 61, 10, 46, 37, 11, 44, 7, 18, 63, 20, 29, 7, 49, 28, 54, 8, 43, 4, 48, 18, 63, 12, 29, 48, 24, 59, 20 },
1014   { 13, 36, 28, 54, 35, 2, 56, 46, 16, 49, 22, 40, 11, 34, 14, 43, 29, 12, 63, 48, 2, 61, 7, 15, 28, 30, 50, 9, 61, 33, 38, 23, 54, 13, 61, 33, 3, 59, 16, 35, 58, 40, 5, 38, 13, 57, 3, 58, 37, 21, 45, 12, 39, 7, 35, 30, 13, 56, 22, 62, 27, 6, 55, 10, 48, 21, 33, 2, 38, 23, 40, 20, 44, 29, 59, 4, 26, 12, 33, 47, 28, 53, 31, 13, 59, 41, 27, 49, 26, 54, 45, 16, 53, 21, 35, 7, 59, 26, 11, 56, 1, 24, 33, 4, 28, 62, 21, 49, 31, 2, 56, 39, 24, 58, 13, 17, 37, 21, 56, 10, 38, 0, 34, 55, 15, 43, 1, 52 },
1015   { 42, 9, 50, 6, 25, 60, 14, 38, 10, 29, 53, 18, 57, 3, 25, 51, 0, 53, 25, 17, 29, 37, 52, 46, 0, 62, 14, 37, 4, 50, 10, 44, 0, 46, 20, 25, 50, 19, 55, 0, 23, 31, 62, 34, 11, 45, 19, 32, 0, 53, 10, 59, 23, 47, 18, 60, 42, 28, 37, 3, 50, 15, 35, 44, 0, 51, 27, 60, 9, 57, 16, 58, 11, 22, 46, 15, 53, 48, 7, 42, 0, 60, 5, 49, 24, 54, 9, 17, 39, 5, 34, 62, 3, 40, 60, 31, 0, 47, 29, 16, 49, 39, 59, 17, 50, 0, 40, 13, 53, 38, 16, 46, 0, 42, 34, 60, 2, 53, 29, 31, 58, 46, 27, 6, 61, 8, 37, 28 },
1016   { 0, 63, 21, 40, 45, 18, 51, 23, 63, 34, 6, 43, 28, 38, 55, 19, 40, 35, 8, 41, 54, 10, 21, 32, 39, 23, 53, 26, 55, 28, 22, 63, 30, 34, 9, 48, 6, 38, 29, 43, 49, 6, 18, 52, 27, 61, 9, 43, 28, 42, 33, 26, 56, 3, 51, 23, 0, 48, 16, 45, 32, 25, 63, 20, 57, 17, 42, 12, 35, 47, 5, 31, 39, 56, 6, 30, 34, 21, 61, 25, 14, 40, 22, 38, 15, 6, 36, 56, 20, 60, 25, 12, 51, 27, 10, 56, 42, 20, 36, 63, 32, 6, 21, 41, 12, 34, 60, 26, 5, 48, 27, 10, 62, 19, 6, 47, 39, 14, 45, 7, 24, 17, 41, 32, 23, 51, 19, 56 },
1017   { 45, 31, 15, 59, 4, 33, 7, 47, 0, 41, 13, 61, 4, 47, 9, 23, 60, 14, 57, 31, 4, 45, 59, 6, 58, 10, 44, 20, 8, 42, 15, 6, 55, 17, 58, 31, 53, 12, 61, 10, 15, 57, 43, 2, 23, 35, 48, 14, 54, 6, 18, 49, 15, 38, 11, 34, 62, 9, 21, 58, 11, 41, 4, 31, 38, 8, 29, 55, 19, 36, 27, 52, 0, 25, 50, 43, 1, 39, 8, 55, 35, 51, 10, 30, 45, 62, 29, 2, 46, 10, 32, 48, 18, 38, 5, 22, 33, 8, 51, 3, 14, 44, 54, 25, 57, 30, 18, 52, 33, 22, 59, 28, 36, 52, 32, 21, 26, 50, 5, 55, 35, 60, 14, 54, 4, 40, 16, 33 },
1018   { 27, 3, 49, 10, 30, 40, 55, 27, 57, 24, 52, 21, 32, 17, 60, 30, 5, 44, 27, 49, 19, 34, 13, 24, 43, 36, 3, 49, 31, 59, 37, 48, 26, 41, 2, 41, 14, 36, 21, 32, 40, 26, 13, 49, 55, 5, 16, 40, 25, 60, 36, 1, 63, 29, 17, 44, 25, 40, 52, 5, 29, 47, 54, 13, 46, 24, 60, 4, 51, 22, 63, 14, 45, 18, 12, 62, 17, 57, 19, 42, 3, 26, 58, 48, 1, 21, 40, 52, 23, 37, 44, 1, 29, 58, 43, 50, 15, 61, 19, 45, 58, 28, 7, 48, 2, 46, 8, 42, 3, 55, 8, 50, 12, 4, 55, 10, 63, 33, 20, 40, 11, 3, 46, 20, 48, 26, 61, 11 },
1019   { 44, 56, 24, 36, 53, 19, 12, 37, 16, 44, 7, 36, 49, 54, 11, 37, 48, 21, 15, 1, 62, 25, 47, 56, 16, 18, 51, 12, 40, 1, 24, 11, 52, 16, 23, 59, 28, 1, 45, 53, 4, 60, 37, 21, 39, 30, 63, 20, 52, 10, 30, 45, 8, 41, 54, 4, 57, 7, 34, 55, 36, 18, 23, 59, 2, 48, 11, 32, 44, 1, 41, 8, 33, 54, 38, 23, 30, 46, 6, 29, 62, 18, 32, 16, 55, 34, 14, 11, 61, 7, 55, 16, 53, 13, 23, 2, 55, 37, 26, 10, 33, 23, 36, 16, 38, 22, 56, 15, 24, 43, 35, 17, 44, 40, 25, 46, 16, 1, 57, 25, 49, 36, 28, 62, 9, 35, 7, 53 },
1020   { 17, 38, 8, 61, 1, 50, 26, 62, 3, 31, 56, 15, 1, 26, 40, 2, 34, 51, 56, 36, 42, 9, 38, 2, 29, 60, 32, 57, 19, 62, 34, 47, 4, 57, 39, 7, 44, 63, 24, 18, 46, 28, 8, 54, 1, 34, 7, 46, 3, 37, 50, 23, 57, 21, 13, 46, 31, 20, 43, 15, 1, 61, 8, 33, 37, 17, 56, 26, 15, 49, 24, 59, 28, 3, 56, 9, 52, 32, 13, 49, 10, 43, 5, 45, 8, 25, 59, 42, 28, 33, 19, 40, 8, 63, 35, 47, 25, 4, 40, 52, 1, 60, 12, 53, 63, 9, 29, 60, 37, 19, 1, 62, 31, 20, 58, 12, 41, 30, 43, 9, 18, 52, 22, 1, 39, 30, 58, 21 },
1021   { 13, 47, 29, 18, 43, 34, 5, 48, 20, 42, 10, 45, 30, 58, 20, 63, 24, 11, 6, 28, 54, 14, 22, 52, 41, 7, 26, 5, 45, 15, 53, 13, 35, 27, 18, 50, 12, 33, 5, 56, 10, 17, 45, 24, 59, 15, 50, 26, 56, 13, 19, 5, 32, 52, 27, 36, 2, 61, 12, 26, 49, 40, 27, 52, 13, 50, 6, 39, 61, 34, 10, 37, 48, 20, 41, 27, 2, 36, 59, 24, 54, 33, 63, 20, 38, 50, 3, 17, 52, 4, 58, 27, 45, 21, 32, 11, 48, 17, 57, 20, 46, 38, 25, 43, 4, 34, 51, 6, 13, 45, 57, 26, 6, 48, 2, 35, 53, 23, 61, 34, 59, 6, 42, 56, 13, 51, 2, 41 },
1022   { 32, 5, 55, 23, 58, 14, 22, 52, 29, 15, 61, 25, 51, 8, 43, 13, 53, 41, 46, 20, 3, 33, 63, 11, 48, 21, 54, 38, 28, 3, 30, 43, 21, 62, 9, 31, 55, 22, 51, 29, 37, 62, 32, 12, 42, 29, 41, 9, 33, 44, 62, 28, 43, 1, 59, 19, 48, 30, 51, 39, 24, 4, 58, 19, 42, 29, 22, 43, 3, 18, 53, 5, 13, 50, 16, 60, 45, 21, 7, 40, 15, 0, 26, 53, 13, 31, 43, 24, 47, 31, 15, 49, 2, 41, 6, 59, 29, 42, 9, 30, 14, 7, 49, 18, 31, 47, 20, 39, 49, 32, 11, 41, 54, 15, 61, 18, 7, 38, 4, 13, 44, 28, 15, 32, 45, 19, 27, 49 },
1023   { 63, 34, 11, 39, 2, 45, 37, 8, 59, 39, 33, 4, 36, 17, 48, 5, 29, 18, 32, 61, 39, 50, 5, 27, 35, 0, 46, 12, 22, 49, 60, 6, 54, 0, 38, 49, 2, 42, 15, 40, 0, 47, 20, 51, 3, 57, 18, 61, 22, 0, 39, 16, 55, 12, 35, 8, 41, 22, 6, 59, 16, 45, 10, 36, 0, 62, 9, 54, 30, 58, 21, 43, 63, 31, 7, 35, 12, 48, 58, 28, 47, 37, 41, 9, 57, 20, 61, 0, 36, 11, 57, 35, 23, 52, 37, 18, 0, 62, 22, 55, 35, 62, 27, 54, 0, 15, 61, 28, 2, 59, 22, 9, 37, 27, 33, 51, 29, 48, 19, 50, 25, 37, 10, 57, 5, 37, 60, 8 },
1024   { 20, 25, 46, 52, 31, 60, 12, 55, 0, 19, 11, 46, 62, 35, 23, 38, 57, 0, 55, 10, 16, 30, 58, 44, 17, 59, 29, 63, 42, 8, 36, 20, 33, 46, 16, 61, 25, 35, 8, 54, 26, 7, 58, 22, 34, 6, 47, 14, 53, 31, 48, 9, 37, 25, 49, 63, 16, 55, 45, 14, 34, 63, 21, 53, 25, 33, 46, 16, 35, 7, 46, 29, 0, 39, 25, 55, 22, 34, 18, 4, 56, 11, 23, 51, 28, 6, 39, 14, 62, 44, 19, 8, 60, 12, 56, 28, 50, 34, 39, 5, 51, 3, 41, 12, 57, 35, 10, 53, 25, 17, 52, 30, 47, 0, 43, 14, 5, 57, 31, 55, 0, 63, 47, 23, 54, 24, 14, 43 },
1025   { 0, 57, 16, 6, 26, 19, 35, 28, 49, 42, 54, 26, 21, 1, 59, 27, 9, 47, 26, 44, 50, 22, 13, 40, 8, 37, 10, 34, 17, 56, 25, 58, 13, 27, 44, 9, 20, 58, 31, 17, 60, 36, 10, 41, 53, 25, 36, 39, 4, 24, 58, 17, 60, 4, 22, 38, 10, 32, 0, 50, 31, 7, 28, 47, 12, 57, 5, 26, 52, 23, 14, 40, 57, 17, 47, 5, 53, 1, 44, 31, 19, 60, 46, 2, 35, 48, 30, 54, 22, 5, 51, 39, 25, 31, 4, 43, 14, 9, 45, 16, 24, 44, 19, 29, 40, 23, 44, 7, 38, 42, 4, 63, 12, 54, 23, 59, 22, 42, 8, 15, 40, 21, 8, 34, 3, 41, 30, 50 },
1026   { 39, 10, 48, 33, 41, 54, 5, 47, 23, 13, 32, 7, 52, 44, 14, 39, 58, 18, 35, 6, 37, 2, 60, 24, 55, 19, 53, 2, 51, 32, 1, 41, 51, 4, 40, 29, 47, 3, 52, 44, 13, 49, 28, 16, 1, 62, 11, 27, 52, 35, 5, 42, 29, 47, 14, 56, 28, 53, 26, 38, 9, 56, 40, 3, 38, 15, 41, 60, 1, 37, 50, 25, 11, 28, 61, 19, 42, 62, 10, 52, 39, 6, 32, 14, 58, 17, 7, 26, 42, 34, 27, 10, 54, 40, 20, 63, 26, 53, 21, 61, 32, 7, 59, 48, 3, 56, 18, 31, 58, 14, 49, 21, 36, 16, 45, 9, 36, 24, 62, 45, 27, 31, 53, 17, 49, 12, 62, 18 },
1027   { 28, 59, 21, 58, 2, 16, 38, 9, 62, 3, 56, 41, 10, 31, 50, 4, 32, 52, 12, 63, 23, 46, 33, 31, 4, 48, 25, 43, 14, 23, 47, 11, 22, 55, 14, 60, 23, 37, 11, 39, 23, 2, 45, 56, 31, 43, 19, 55, 16, 46, 21, 51, 11, 33, 44, 2, 41, 18, 5, 52, 23, 44, 17, 60, 27, 49, 11, 32, 44, 10, 54, 2, 56, 33, 8, 38, 13, 29, 36, 16, 24, 63, 27, 51, 21, 43, 56, 12, 49, 3, 59, 48, 1, 15, 46, 7, 36, 2, 47, 11, 50, 27, 37, 13, 33, 8, 51, 46, 1, 34, 28, 40, 3, 33, 60, 29, 47, 1, 35, 11, 59, 42, 2, 60, 26, 46, 6, 35 },
1028   { 4, 43, 9, 29, 36, 63, 24, 44, 20, 50, 30, 17, 60, 22, 16, 43, 25, 3, 42, 19, 51, 15, 8, 54, 42, 15, 61, 5, 39, 57, 18, 61, 31, 48, 34, 2, 50, 19, 57, 5, 63, 33, 19, 38, 13, 27, 48, 7, 32, 61, 2, 26, 58, 6, 24, 50, 13, 61, 42, 20, 62, 2, 35, 20, 51, 4, 62, 18, 23, 58, 20, 31, 43, 15, 51, 45, 26, 50, 4, 55, 45, 3, 35, 9, 38, 1, 32, 61, 20, 45, 17, 33, 24, 57, 29, 51, 22, 58, 38, 30, 15, 1, 54, 21, 63, 43, 26, 12, 24, 56, 8, 60, 50, 19, 5, 52, 13, 54, 17, 50, 4, 16, 36, 12, 32, 56, 22, 54 },
1029   { 51, 25, 40, 53, 12, 49, 15, 57, 34, 7, 38, 47, 2, 36, 55, 8, 61, 30, 56, 7, 28, 59, 48, 11, 27, 35, 21, 45, 28, 36, 9, 38, 6, 16, 24, 63, 10, 32, 28, 43, 21, 53, 5, 60, 8, 57, 3, 45, 11, 37, 15, 54, 40, 20, 62, 36, 27, 34, 11, 48, 30, 15, 54, 8, 30, 42, 22, 34, 48, 13, 35, 63, 4, 37, 22, 2, 59, 9, 41, 23, 13, 41, 49, 18, 59, 24, 40, 5, 37, 30, 9, 61, 44, 6, 37, 11, 33, 17, 5, 55, 41, 60, 23, 39, 17, 5, 30, 62, 41, 16, 46, 25, 11, 56, 39, 26, 20, 38, 29, 39, 22, 52, 44, 20, 48, 1, 38, 14 },
1030   { 15, 33, 2, 18, 44, 6, 27, 0, 32, 61, 25, 12, 58, 28, 40, 20, 47, 13, 34, 43, 38, 1, 23, 62, 40, 0, 51, 10, 63, 3, 52, 26, 44, 30, 45, 6, 41, 54, 0, 51, 12, 30, 46, 24, 49, 22, 40, 33, 63, 23, 43, 30, 9, 47, 0, 17, 54, 7, 57, 3, 37, 47, 24, 46, 13, 55, 7, 52, 2, 42, 6, 26, 49, 18, 60, 34, 16, 57, 33, 20, 61, 30, 8, 54, 14, 46, 12, 53, 16, 55, 38, 13, 22, 53, 18, 59, 46, 27, 43, 19, 32, 10, 45, 6, 49, 36, 52, 2, 20, 55, 6, 39, 32, 15, 44, 3, 58, 10, 63, 6, 56, 30, 7, 58, 9, 40, 19, 63 },
1031   { 10, 47, 61, 23, 55, 31, 52, 42, 17, 45, 4, 51, 27, 6, 15, 53, 0, 49, 26, 10, 56, 18, 36, 6, 20, 58, 32, 30, 13, 49, 19, 56, 0, 59, 12, 53, 27, 17, 38, 25, 48, 9, 15, 36, 14, 30, 59, 17, 0, 50, 8, 58, 18, 56, 31, 45, 21, 41, 29, 19, 60, 6, 32, 59, 0, 36, 29, 39, 19, 59, 46, 12, 55, 30, 10, 47, 24, 3, 28, 48, 0, 55, 44, 27, 33, 4, 63, 29, 49, 0, 26, 50, 34, 2, 42, 14, 0, 62, 9, 56, 3, 52, 28, 34, 58, 9, 20, 48, 37, 32, 22, 53, 0, 62, 27, 49, 34, 46, 21, 33, 41, 14, 25, 37, 53, 29, 31, 45 },
1032   { 56, 28, 7, 37, 11, 36, 20, 9, 54, 14, 39, 19, 34, 63, 45, 37, 24, 17, 60, 31, 21, 45, 53, 29, 47, 15, 7, 55, 40, 23, 34, 14, 42, 20, 37, 35, 15, 59, 7, 62, 34, 40, 59, 1, 51, 42, 10, 28, 54, 21, 35, 5, 38, 13, 36, 4, 59, 12, 39, 53, 15, 43, 9, 21, 39, 62, 16, 56, 25, 9, 32, 38, 0, 41, 14, 51, 40, 53, 43, 11, 37, 17, 5, 22, 57, 39, 19, 7, 42, 21, 60, 10, 31, 63, 25, 52, 30, 49, 36, 25, 48, 17, 61, 14, 22, 42, 29, 13, 60, 11, 47, 18, 35, 41, 7, 23, 4, 16, 51, 11, 0, 48, 61, 3, 17, 50, 5, 24 },
1033   { 0, 42, 21, 49, 60, 3, 57, 40, 29, 48, 23, 56, 42, 11, 22, 5, 59, 39, 4, 50, 3, 41, 12, 57, 25, 50, 44, 18, 4, 46, 7, 62, 33, 50, 4, 56, 21, 32, 43, 18, 3, 23, 55, 34, 20, 4, 53, 38, 12, 46, 29, 52, 25, 61, 23, 51, 26, 46, 1, 34, 25, 57, 28, 51, 26, 11, 50, 3, 44, 28, 53, 21, 57, 27, 62, 6, 31, 19, 8, 63, 26, 59, 36, 47, 15, 29, 50, 25, 35, 47, 18, 41, 4, 48, 8, 40, 12, 23, 6, 44, 13, 40, 1, 31, 55, 0, 61, 43, 4, 50, 26, 58, 9, 53, 24, 61, 42, 55, 31, 43, 57, 20, 34, 27, 43, 8, 59, 39 },
1034   { 18, 51, 30, 13, 26, 16, 46, 22, 2, 59, 8, 30, 1, 48, 33, 51, 29, 9, 46, 16, 62, 14, 33, 2, 38, 9, 27, 60, 37, 26, 53, 17, 28, 10, 24, 46, 2, 49, 8, 57, 29, 45, 6, 26, 62, 44, 18, 25, 61, 3, 42, 14, 49, 10, 43, 6, 17, 32, 63, 10, 49, 4, 40, 14, 45, 33, 22, 37, 12, 61, 5, 17, 43, 7, 23, 37, 15, 58, 49, 13, 39, 21, 10, 52, 1, 62, 9, 56, 12, 2, 58, 28, 36, 16, 56, 28, 56, 35, 20, 63, 24, 37, 51, 8, 45, 25, 16, 33, 27, 38, 2, 44, 13, 30, 17, 36, 12, 26, 5, 18, 28, 47, 13, 60, 23, 45, 13, 33 },
1035   { 55, 4, 62, 34, 52, 38, 7, 63, 32, 37, 13, 53, 25, 62, 18, 12, 55, 41, 27, 35, 24, 49, 31, 52, 17, 63, 34, 1, 56, 12, 41, 2, 48, 58, 39, 16, 61, 27, 41, 52, 13, 19, 50, 39, 11, 31, 57, 6, 32, 40, 20, 55, 1, 28, 33, 57, 48, 8, 37, 22, 44, 18, 53, 1, 61, 5, 54, 16, 47, 36, 50, 24, 55, 34, 48, 45, 1, 30, 33, 46, 2, 50, 32, 42, 25, 34, 43, 21, 38, 52, 23, 45, 14, 54, 21, 4, 44, 16, 53, 29, 10, 47, 19, 57, 12, 54, 39, 10, 51, 15, 63, 21, 57, 40, 51, 1, 48, 57, 37, 62, 2, 38, 9, 52, 1, 35, 58, 22 },
1036   { 36, 46, 10, 42, 1, 27, 43, 15, 50, 21, 45, 16, 41, 3, 35, 44, 20, 1, 57, 11, 55, 7, 43, 8, 22, 42, 13, 46, 21, 39, 31, 60, 22, 5, 29, 44, 11, 35, 20, 4, 36, 58, 32, 15, 47, 2, 36, 48, 16, 60, 8, 35, 44, 63, 16, 2, 40, 26, 55, 14, 58, 35, 24, 31, 19, 42, 31, 58, 1, 29, 10, 40, 2, 19, 12, 54, 22, 61, 7, 24, 56, 5, 28, 16, 54, 3, 15, 58, 6, 30, 8, 62, 1, 43, 31, 47, 7, 59, 1, 38, 58, 4, 34, 27, 38, 5, 31, 59, 7, 46, 30, 3, 34, 6, 28, 59, 20, 8, 32, 15, 53, 24, 55, 31, 19, 49, 11, 26 },
1037   { 2, 24, 16, 58, 19, 55, 5, 35, 10, 61, 4, 28, 57, 24, 58, 7, 31, 47, 22, 38, 19, 28, 61, 36, 54, 5, 59, 29, 6, 52, 15, 11, 43, 36, 8, 54, 52, 1, 62, 25, 47, 9, 1, 60, 28, 53, 24, 14, 46, 27, 51, 22, 12, 24, 38, 53, 20, 11, 51, 3, 29, 7, 48, 63, 8, 49, 9, 21, 52, 14, 63, 32, 46, 60, 35, 4, 41, 16, 52, 35, 18, 42, 59, 7, 36, 61, 45, 27, 33, 51, 19, 39, 34, 11, 61, 18, 33, 41, 28, 15, 54, 22, 42, 3, 49, 21, 47, 18, 36, 23, 55, 19, 48, 24, 45, 10, 33, 44, 50, 40, 7, 35, 15, 41, 63, 6, 40, 54 },
1038   { 62, 41, 32, 8, 47, 28, 60, 24, 44, 30, 38, 49, 9, 33, 14, 40, 50, 14, 60, 2, 54, 40, 0, 20, 25, 39, 16, 49, 24, 35, 57, 47, 19, 61, 33, 18, 23, 37, 13, 55, 31, 43, 22, 41, 17, 8, 42, 58, 0, 37, 5, 56, 31, 54, 7, 30, 60, 33, 42, 17, 59, 39, 12, 27, 38, 17, 35, 41, 27, 45, 20, 7, 25, 15, 29, 58, 27, 47, 11, 40, 14, 54, 23, 46, 19, 31, 11, 40, 13, 49, 5, 58, 24, 51, 26, 6, 50, 20, 49, 9, 32, 46, 17, 60, 14, 63, 24, 1, 57, 41, 9, 43, 14, 62, 16, 52, 3, 27, 14, 22, 61, 45, 4, 28, 9, 47, 29, 17 },
1039   { 5, 50, 12, 53, 38, 18, 11, 51, 0, 55, 17, 6, 47, 54, 19, 63, 5, 26, 34, 45, 13, 30, 47, 58, 10, 48, 32, 3, 62, 9, 26, 0, 25, 14, 50, 3, 47, 30, 42, 16, 6, 63, 12, 49, 33, 55, 21, 10, 34, 63, 18, 41, 3, 47, 19, 43, 0, 49, 8, 28, 46, 20, 52, 0, 56, 24, 60, 3, 59, 5, 39, 57, 48, 52, 9, 38, 3, 21, 26, 60, 0, 32, 12, 38, 4, 48, 53, 0, 60, 15, 29, 44, 18, 10, 38, 57, 13, 60, 2, 26, 62, 7, 50, 29, 35, 8, 40, 53, 28, 12, 60, 33, 38, 5, 37, 29, 60, 39, 56, 0, 30, 18, 50, 34, 59, 25, 14, 44 },
1040   { 20, 31, 60, 22, 3, 49, 33, 25, 40, 13, 34, 59, 22, 36, 0, 28, 37, 56, 8, 18, 51, 16, 4, 45, 27, 12, 53, 42, 18, 44, 51, 31, 55, 40, 28, 58, 7, 60, 10, 51, 27, 37, 24, 56, 5, 26, 44, 29, 50, 23, 45, 11, 34, 15, 59, 27, 13, 23, 62, 37, 4, 57, 15, 32, 42, 6, 47, 11, 30, 43, 23, 13, 0, 36, 18, 44, 63, 51, 37, 29, 49, 20, 57, 27, 62, 9, 24, 35, 23, 53, 37, 3, 42, 55, 0, 36, 23, 39, 31, 43, 17, 37, 24, 11, 52, 43, 19, 32, 5, 50, 26, 0, 56, 21, 54, 11, 19, 6, 47, 25, 59, 42, 12, 54, 21, 3, 38, 57 },
1041   { 48, 0, 35, 27, 44, 14, 59, 7, 57, 46, 26, 2, 42, 12, 52, 43, 10, 27, 53, 42, 32, 62, 37, 21, 34, 61, 7, 23, 36, 4, 38, 12, 41, 5, 17, 45, 22, 27, 39, 21, 59, 0, 45, 18, 39, 62, 3, 38, 14, 7, 54, 26, 61, 39, 9, 52, 45, 36, 18, 50, 10, 34, 44, 22, 50, 14, 36, 55, 17, 34, 53, 62, 33, 26, 56, 6, 31, 12, 6, 53, 9, 44, 2, 50, 20, 40, 55, 17, 47, 7, 26, 63, 22, 32, 48, 16, 46, 8, 52, 12, 57, 41, 0, 56, 25, 3, 61, 14, 45, 35, 18, 44, 12, 46, 23, 42, 32, 51, 35, 10, 17, 36, 23, 1, 45, 52, 32, 10 },
1042   { 37, 15, 43, 8, 63, 39, 21, 31, 16, 37, 19, 62, 30, 46, 17, 60, 21, 48, 1, 23, 6, 25, 11, 56, 1, 40, 30, 58, 15, 54, 21, 59, 9, 63, 35, 56, 11, 51, 2, 46, 34, 14, 53, 7, 30, 11, 51, 19, 60, 40, 30, 1, 24, 50, 20, 32, 3, 56, 5, 25, 31, 13, 61, 2, 29, 60, 25, 20, 51, 2, 27, 8, 18, 42, 10, 45, 21, 34, 43, 17, 62, 29, 41, 14, 34, 6, 30, 43, 2, 57, 33, 13, 45, 12, 27, 62, 4, 55, 21, 35, 5, 27, 45, 33, 16, 47, 30, 54, 22, 10, 51, 27, 63, 7, 49, 1, 58, 22, 15, 43, 53, 7, 57, 39, 27, 12, 61, 24 },
1043   { 56, 51, 26, 56, 19, 2, 41, 54, 5, 52, 9, 48, 6, 23, 39, 4, 32, 15, 63, 35, 59, 49, 43, 15, 52, 19, 50, 9, 46, 33, 1, 29, 48, 20, 32, 1, 38, 33, 19, 54, 9, 32, 24, 48, 58, 35, 16, 48, 4, 52, 13, 57, 33, 5, 45, 59, 15, 29, 41, 55, 47, 39, 23, 53, 9, 40, 4, 57, 10, 44, 48, 40, 50, 14, 61, 24, 55, 1, 59, 22, 33, 8, 51, 25, 58, 46, 11, 59, 20, 41, 17, 51, 6, 56, 35, 25, 42, 30, 15, 58, 48, 18, 61, 9, 58, 39, 13, 2, 37, 59, 40, 2, 31, 16, 34, 41, 8, 30, 62, 3, 29, 48, 33, 5, 63, 16, 41, 7 },
1044   { 22, 4, 46, 11, 33, 51, 29, 10, 62, 24, 43, 27, 15, 58, 50, 25, 54, 44, 9, 38, 18, 3, 29, 57, 32, 5, 26, 43, 17, 61, 24, 52, 8, 42, 23, 53, 15, 61, 7, 28, 57, 43, 4, 40, 20, 2, 43, 25, 32, 35, 21, 43, 17, 48, 10, 22, 38, 54, 11, 21, 1, 58, 16, 30, 48, 18, 46, 32, 38, 13, 22, 4, 59, 35, 2, 51, 30, 39, 15, 47, 4, 56, 13, 37, 1, 28, 16, 52, 32, 9, 61, 29, 38, 19, 3, 52, 10, 48, 1, 32, 11, 40, 20, 36, 6, 22, 49, 29, 55, 6, 20, 56, 36, 52, 19, 60, 26, 46, 18, 54, 40, 13, 20, 46, 35, 19, 49, 29 },
1045   { 61, 17, 34, 53, 23, 6, 48, 35, 20, 40, 1, 56, 36, 29, 11, 34, 7, 41, 14, 30, 55, 20, 46, 8, 24, 38, 63, 2, 37, 10, 45, 14, 34, 49, 6, 13, 44, 25, 49, 41, 21, 12, 61, 15, 54, 29, 63, 12, 56, 8, 49, 2, 62, 36, 28, 61, 0, 25, 41, 63, 35, 8, 44, 6, 37, 62, 7, 21, 63, 28, 55, 31, 16, 24, 41, 19, 9, 57, 27, 36, 18, 42, 31, 62, 22, 55, 38, 4, 27, 47, 1, 40, 14, 54, 43, 20, 60, 23, 38, 63, 25, 51, 2, 53, 26, 63, 10, 42, 17, 34, 47, 25, 13, 5, 44, 11, 55, 2, 38, 27, 6, 60, 52, 25, 9, 55, 1, 40 },
1046   { 8, 30, 58, 3, 42, 61, 17, 38, 13, 59, 32, 10, 54, 3, 51, 20, 61, 26, 57, 2, 46, 33, 12, 60, 41, 13, 48, 29, 55, 20, 39, 27, 57, 18, 62, 29, 55, 2, 31, 16, 37, 50, 26, 36, 6, 46, 9, 41, 27, 57, 23, 39, 26, 6, 51, 12, 31, 46, 7, 16, 27, 52, 19, 56, 26, 12, 33, 53, 1, 41, 8, 57, 46, 7, 54, 32, 47, 5, 49, 11, 60, 23, 5, 48, 10, 43, 19, 63, 35, 24, 49, 21, 59, 5, 31, 37, 14, 44, 7, 42, 6, 30, 46, 13, 44, 32, 19, 50, 4, 58, 8, 30, 62, 38, 28, 53, 21, 36, 13, 50, 21, 33, 15, 2, 44, 31, 14, 47 },
1047   { 37, 13, 39, 16, 28, 9, 57, 0, 25, 49, 21, 45, 18, 47, 12, 42, 0, 49, 22, 39, 16, 53, 25, 36, 0, 52, 22, 16, 6, 60, 4, 51, 0, 26, 37, 47, 10, 36, 63, 5, 57, 0, 18, 59, 23, 33, 51, 19, 0, 44, 15, 11, 54, 17, 42, 35, 53, 18, 58, 33, 49, 4, 34, 42, 0, 50, 43, 25, 16, 49, 34, 20, 37, 28, 12, 63, 16, 38, 25, 44, 0, 40, 52, 17, 35, 3, 50, 14, 8, 53, 11, 36, 25, 45, 9, 62, 0, 54, 28, 17, 50, 55, 15, 24, 57, 0, 53, 34, 23, 41, 15, 45, 0, 49, 16, 4, 48, 9, 63, 45, 0, 42, 58, 37, 61, 22, 54, 26 },
1048   { 0, 50, 21, 47, 54, 36, 27, 45, 52, 4, 34, 15, 63, 29, 37, 59, 17, 31, 6, 61, 28, 5, 48, 18, 59, 27, 34, 56, 44, 31, 35, 12, 41, 59, 16, 3, 40, 20, 50, 22, 30, 40, 52, 10, 45, 3, 59, 22, 37, 61, 29, 46, 31, 58, 2, 22, 9, 43, 3, 39, 14, 61, 24, 54, 15, 29, 11, 60, 39, 17, 5, 61, 0, 44, 50, 3, 31, 14, 58, 21, 54, 28, 15, 45, 60, 26, 33, 58, 44, 22, 60, 2, 57, 34, 49, 27, 18, 34, 21, 59, 29, 4, 36, 41, 8, 39, 28, 11, 62, 26, 53, 20, 35, 24, 59, 32, 29, 39, 24, 31, 57, 23, 11, 28, 5, 36, 11, 59 },
1049   { 44, 32, 63, 5, 20, 12, 41, 7, 30, 61, 42, 8, 39, 5, 33, 8, 24, 53, 45, 11, 37, 58, 7, 44, 10, 50, 3, 40, 8, 22, 53, 19, 46, 9, 33, 52, 24, 58, 8, 44, 13, 47, 8, 34, 38, 30, 14, 47, 7, 34, 4, 55, 9, 19, 40, 49, 56, 26, 60, 21, 30, 45, 10, 19, 40, 58, 23, 36, 3, 52, 45, 23, 54, 13, 22, 42, 53, 45, 7, 33, 10, 36, 57, 6, 29, 12, 41, 0, 30, 15, 41, 30, 17, 7, 16, 53, 40, 56, 2, 39, 12, 61, 10, 52, 31, 60, 16, 45, 1, 37, 7, 61, 40, 10, 43, 17, 58, 7, 54, 14, 4, 51, 39, 49, 18, 56, 42, 20 },
1050   { 14, 6, 24, 36, 56, 49, 22, 60, 18, 14, 23, 51, 26, 57, 21, 52, 41, 14, 35, 50, 19, 31, 40, 23, 33, 14, 63, 17, 32, 47, 7, 62, 23, 30, 56, 11, 42, 27, 14, 60, 35, 19, 28, 61, 17, 55, 25, 39, 53, 17, 42, 21, 38, 63, 25, 5, 14, 36, 12, 50, 1, 37, 59, 32, 2, 51, 6, 56, 27, 32, 11, 30, 38, 26, 60, 8, 26, 19, 62, 39, 50, 2, 21, 39, 53, 23, 56, 19, 49, 39, 5, 46, 55, 23, 42, 4, 31, 11, 47, 26, 45, 22, 48, 18, 21, 5, 48, 25, 57, 14, 47, 30, 3, 56, 12, 50, 1, 42, 19, 47, 35, 17, 8, 30, 45, 25, 4, 51 },
1051   { 28, 58, 43, 1, 31, 8, 33, 2, 44, 55, 32, 1, 60, 12, 46, 27, 4, 62, 23, 1, 56, 13, 62, 2, 54, 36, 25, 51, 1, 57, 26, 42, 3, 49, 17, 38, 1, 48, 31, 4, 54, 3, 50, 24, 1, 49, 5, 63, 13, 27, 52, 1, 48, 13, 45, 33, 52, 30, 46, 20, 55, 28, 6, 48, 24, 38, 20, 47, 14, 62, 48, 9, 58, 4, 36, 30, 56, 1, 34, 12, 18, 63, 25, 48, 4, 16, 37, 7, 62, 10, 52, 28, 13, 50, 36, 63, 24, 51, 15, 58, 8, 33, 1, 38, 56, 35, 42, 9, 33, 51, 22, 18, 48, 32, 27, 37, 23, 61, 33, 11, 59, 29, 62, 1, 53, 10, 60, 33 },
1052   { 12, 39, 17, 52, 26, 46, 53, 38, 25, 11, 48, 36, 16, 43, 2, 35, 55, 17, 39, 29, 43, 9, 28, 45, 20, 5, 46, 12, 42, 28, 13, 52, 36, 6, 60, 22, 54, 17, 62, 39, 25, 42, 15, 55, 44, 20, 31, 10, 35, 57, 24, 32, 29, 6, 59, 18, 7, 62, 3, 41, 10, 44, 16, 54, 13, 62, 31, 9, 41, 1, 21, 43, 18, 47, 15, 40, 11, 49, 28, 55, 46, 30, 8, 43, 32, 61, 28, 47, 25, 34, 21, 61, 32, 1, 20, 9, 46, 6, 35, 19, 41, 54, 27, 63, 14, 3, 51, 20, 62, 2, 38, 55, 8, 21, 63, 6, 46, 9, 26, 51, 3, 24, 43, 34, 16, 41, 18, 48 },
1053   { 62, 23, 55, 9, 15, 62, 19, 13, 58, 40, 6, 30, 54, 19, 50, 31, 10, 44, 6, 59, 21, 47, 51, 15, 60, 39, 30, 54, 21, 61, 19, 33, 14, 29, 43, 11, 34, 45, 7, 21, 10, 56, 36, 6, 38, 11, 58, 42, 2, 47, 11, 60, 50, 16, 41, 28, 38, 23, 47, 17, 35, 63, 22, 33, 42, 5, 45, 17, 53, 35, 25, 56, 33, 6, 51, 19, 60, 23, 43, 15, 5, 40, 58, 13, 51, 1, 45, 11, 54, 3, 43, 8, 37, 48, 59, 29, 39, 21, 61, 43, 3, 31, 10, 44, 24, 29, 60, 12, 28, 40, 11, 25, 43, 52, 14, 41, 16, 57, 44, 20, 40, 55, 12, 21, 57, 27, 35, 2 },
1054   { 37, 6, 31, 42, 40, 4, 29, 50, 0, 20, 63, 28, 9, 58, 14, 24, 63, 26, 48, 16, 34, 4, 32, 38, 23, 11, 58, 4, 37, 9, 45, 5, 63, 48, 26, 57, 2, 28, 32, 51, 46, 29, 13, 62, 27, 46, 28, 18, 50, 15, 40, 4, 19, 34, 54, 0, 53, 9, 26, 58, 28, 5, 49, 0, 57, 27, 19, 60, 29, 8, 59, 12, 37, 63, 24, 46, 3, 37, 6, 52, 26, 32, 20, 36, 9, 22, 59, 18, 35, 51, 14, 57, 17, 24, 12, 44, 56, 0, 30, 13, 59, 20, 49, 17, 54, 43, 6, 34, 46, 17, 58, 36, 0, 34, 29, 54, 25, 2, 36, 15, 60, 6, 37, 46, 4, 50, 9, 45 },
1055   { 19, 59, 48, 3, 24, 60, 44, 22, 34, 51, 15, 45, 41, 5, 33, 47, 0, 37, 12, 55, 25, 54, 8, 57, 0, 47, 18, 34, 49, 15, 55, 24, 40, 20, 8, 35, 53, 13, 41, 18, 0, 59, 22, 33, 4, 52, 8, 60, 24, 36, 31, 56, 45, 26, 10, 43, 15, 56, 36, 4, 51, 14, 39, 30, 12, 55, 36, 2, 39, 49, 4, 44, 17, 0, 32, 13, 53, 35, 59, 17, 62, 0, 55, 24, 52, 38, 31, 6, 42, 19, 29, 40, 4, 54, 33, 5, 16, 27, 52, 37, 23, 55, 7, 37, 0, 39, 23, 49, 4, 53, 31, 15, 59, 10, 50, 4, 60, 34, 48, 7, 31, 49, 27, 14, 62, 22, 53, 29 },
1056   { 46, 21, 14, 51, 36, 17, 7, 57, 10, 32, 3, 37, 22, 60, 39, 18, 56, 20, 42, 3, 36, 10, 44, 26, 41, 29, 53, 27, 2, 39, 30, 52, 0, 59, 15, 48, 23, 61, 6, 58, 37, 12, 40, 49, 16, 39, 20, 44, 0, 62, 8, 21, 3, 59, 23, 32, 49, 31, 12, 44, 22, 59, 18, 50, 24, 7, 43, 52, 15, 23, 41, 26, 51, 28, 55, 39, 21, 27, 10, 42, 12, 45, 27, 47, 3, 15, 63, 26, 55, 0, 60, 26, 45, 18, 62, 38, 58, 49, 8, 47, 4, 33, 46, 29, 57, 13, 56, 16, 59, 21, 5, 47, 23, 39, 18, 44, 13, 22, 28, 53, 19, 0, 58, 32, 41, 7, 26, 13 },
1057   { 0, 56, 34, 28, 11, 55, 31, 47, 26, 41, 56, 13, 53, 28, 11, 49, 7, 52, 32, 61, 50, 22, 63, 17, 13, 56, 7, 19, 43, 62, 10, 21, 37, 32, 43, 4, 38, 19, 44, 25, 31, 54, 5, 23, 61, 30, 53, 12, 35, 22, 43, 53, 37, 48, 7, 62, 20, 2, 61, 41, 8, 34, 47, 9, 63, 34, 28, 10, 55, 33, 14, 57, 7, 47, 9, 61, 4, 49, 31, 50, 21, 38, 8, 16, 57, 44, 33, 5, 49, 36, 12, 50, 7, 34, 10, 25, 2, 22, 36, 15, 26, 61, 18, 9, 22, 46, 32, 8, 27, 37, 44, 30, 55, 3, 62, 24, 38, 56, 5, 45, 38, 24, 43, 10, 19, 54, 39, 61 },
1058   { 41, 30, 8, 63, 43, 23, 38, 3, 62, 19, 8, 49, 25, 1, 58, 30, 23, 40, 9, 28, 18, 40, 6, 38, 49, 22, 35, 59, 8, 27, 50, 5, 56, 17, 11, 50, 30, 9, 55, 2, 51, 19, 34, 47, 9, 41, 6, 26, 48, 57, 14, 28, 17, 12, 39, 13, 37, 46, 25, 19, 54, 27, 1, 37, 16, 45, 20, 60, 1, 48, 20, 38, 31, 22, 42, 15, 19, 44, 1, 61, 6, 34, 56, 40, 29, 10, 20, 46, 13, 22, 41, 23, 59, 42, 30, 51, 45, 13, 63, 53, 42, 12, 51, 38, 62, 2, 26, 41, 50, 1, 61, 10, 19, 42, 31, 8, 49, 32, 12, 63, 9, 52, 16, 56, 36, 2, 31, 16 },
1059   { 52, 5, 47, 20, 1, 53, 12, 50, 16, 35, 43, 21, 33, 43, 16, 44, 3, 59, 14, 46, 1, 30, 60, 33, 2, 45, 12, 42, 31, 47, 14, 33, 46, 25, 55, 27, 60, 36, 16, 42, 14, 46, 26, 1, 55, 15, 63, 32, 2, 38, 5, 47, 33, 61, 30, 52, 4, 57, 6, 38, 11, 43, 61, 24, 52, 3, 31, 22, 42, 10, 62, 3, 59, 11, 35, 57, 33, 54, 24, 14, 29, 48, 18, 2, 60, 41, 53, 24, 32, 62, 3, 53, 15, 1, 55, 17, 32, 40, 6, 31, 1, 40, 28, 5, 35, 52, 19, 63, 13, 33, 17, 41, 52, 26, 15, 57, 1, 20, 42, 17, 35, 27, 48, 5, 25, 50, 44, 11 },
1060   { 35, 25, 38, 57, 33, 17, 40, 6, 59, 27, 54, 5, 61, 10, 52, 26, 36, 19, 51, 35, 57, 48, 11, 20, 54, 25, 61, 16, 1, 58, 24, 61, 3, 39, 7, 47, 1, 22, 49, 28, 63, 10, 58, 32, 17, 36, 45, 19, 51, 29, 59, 10, 50, 1, 23, 42, 18, 29, 51, 21, 56, 32, 14, 5, 40, 58, 47, 13, 54, 35, 29, 45, 18, 52, 26, 2, 38, 8, 46, 36, 58, 11, 52, 35, 17, 28, 1, 58, 9, 39, 17, 28, 37, 48, 20, 9, 57, 24, 50, 19, 58, 16, 48, 25, 43, 11, 35, 6, 45, 24, 56, 4, 36, 7, 47, 35, 52, 28, 59, 30, 2, 61, 21, 33, 63, 12, 18, 59 },
1061   { 3, 49, 15, 10, 27, 61, 25, 45, 30, 0, 14, 47, 31, 38, 17, 62, 7, 55, 27, 4, 15, 24, 42, 52, 10, 34, 5, 51, 36, 18, 41, 11, 35, 21, 62, 13, 33, 57, 8, 35, 5, 40, 21, 43, 52, 3, 24, 56, 11, 16, 33, 25, 41, 20, 55, 8, 60, 35, 15, 48, 2, 57, 30, 49, 18, 25, 6, 39, 17, 57, 7, 25, 43, 5, 49, 16, 62, 22, 55, 4, 25, 43, 23, 7, 50, 11, 37, 48, 14, 51, 33, 57, 7, 27, 39, 46, 4, 29, 11, 43, 34, 56, 7, 60, 20, 54, 30, 57, 22, 49, 9, 33, 54, 14, 63, 23, 6, 43, 10, 40, 50, 13, 44, 8, 38, 33, 46, 23 },
1062   { 55, 39, 22, 50, 44, 4, 36, 9, 52, 23, 37, 59, 21, 2, 46, 13, 31, 41, 11, 45, 62, 29, 6, 37, 19, 48, 30, 23, 44, 7, 53, 28, 54, 16, 41, 29, 44, 18, 52, 24, 60, 15, 48, 7, 27, 59, 9, 34, 42, 54, 7, 63, 4, 46, 31, 27, 45, 0, 40, 26, 34, 17, 37, 10, 53, 29, 36, 50, 2, 27, 51, 11, 61, 37, 23, 41, 30, 7, 18, 50, 39, 14, 63, 32, 45, 61, 19, 30, 25, 44, 2, 47, 23, 63, 11, 34, 59, 37, 60, 3, 22, 14, 44, 30, 15, 0, 47, 15, 3, 38, 61, 20, 27, 45, 11, 39, 51, 16, 55, 3, 22, 54, 29, 58, 1, 57, 6, 29 },
1063   { 9, 17, 60, 2, 34, 56, 20, 62, 39, 12, 49, 6, 29, 56, 34, 48, 0, 58, 22, 38, 18, 43, 56, 0, 63, 14, 55, 3, 59, 31, 15, 45, 0, 49, 6, 58, 3, 38, 12, 45, 0, 37, 29, 57, 13, 39, 30, 49, 0, 23, 44, 36, 16, 57, 13, 54, 11, 24, 63, 9, 53, 7, 62, 42, 0, 59, 15, 23, 63, 34, 40, 16, 32, 0, 53, 12, 48, 28, 59, 33, 0, 53, 9, 27, 3, 22, 54, 5, 56, 9, 61, 13, 42, 14, 52, 19, 0, 21, 47, 27, 53, 36, 3, 50, 39, 58, 25, 40, 53, 28, 12, 50, 0, 59, 32, 2, 21, 34, 26, 46, 37, 7, 18, 47, 24, 14, 53, 42 },
1064   { 61, 32, 13, 54, 29, 7, 46, 13, 28, 57, 18, 41, 53, 15, 9, 39, 24, 49, 33, 3, 53, 9, 26, 32, 40, 28, 46, 39, 25, 9, 56, 21, 63, 37, 26, 22, 51, 27, 17, 56, 31, 53, 4, 43, 22, 46, 12, 18, 60, 40, 20, 26, 50, 21, 39, 5, 49, 33, 16, 44, 22, 46, 20, 32, 24, 45, 8, 43, 12, 46, 4, 48, 56, 20, 29, 58, 3, 40, 10, 42, 31, 21, 47, 41, 56, 38, 15, 42, 36, 27, 20, 33, 55, 3, 26, 44, 31, 54, 12, 35, 9, 63, 28, 10, 21, 32, 9, 60, 17, 8, 43, 29, 40, 16, 36, 48, 60, 7, 57, 14, 62, 31, 42, 15, 36, 40, 20, 26 },
1065   { 0, 37, 47, 23, 41, 18, 32, 48, 1, 35, 8, 25, 4, 26, 63, 20, 54, 8, 16, 61, 35, 23, 51, 15, 58, 7, 12, 20, 50, 34, 42, 4, 38, 10, 32, 47, 8, 60, 41, 20, 9, 25, 50, 19, 62, 1, 37, 56, 28, 8, 53, 11, 3, 58, 34, 43, 19, 60, 38, 4, 58, 31, 3, 51, 11, 55, 38, 30, 21, 58, 19, 26, 9, 44, 36, 13, 46, 20, 62, 24, 13, 60, 5, 28, 12, 34, 7, 59, 0, 53, 45, 6, 38, 30, 50, 7, 62, 16, 41, 5, 46, 18, 55, 42, 51, 5, 45, 23, 34, 48, 19, 58, 5, 25, 54, 19, 13, 41, 28, 21, 0, 49, 10, 60, 4, 51, 9, 45 },
1066   { 19, 28, 6, 58, 10, 51, 4, 22, 55, 42, 60, 45, 34, 51, 42, 5, 30, 45, 27, 40, 13, 47, 4, 49, 21, 38, 60, 29, 2, 57, 17, 27, 52, 19, 61, 14, 30, 34, 2, 44, 63, 33, 11, 35, 16, 51, 25, 6, 14, 47, 31, 61, 37, 29, 18, 8, 52, 2, 28, 54, 13, 41, 15, 62, 35, 18, 2, 60, 6, 33, 41, 61, 31, 6, 56, 17, 34, 50, 6, 52, 44, 35, 16, 51, 59, 24, 48, 18, 31, 40, 16, 49, 21, 60, 17, 39, 10, 49, 32, 57, 24, 39, 1, 25, 18, 62, 37, 12, 56, 1, 37, 11, 52, 44, 9, 30, 47, 4, 51, 40, 55, 25, 34, 27, 56, 30, 32, 54 },
1067   { 63, 40, 49, 15, 43, 26, 63, 38, 16, 20, 30, 12, 57, 14, 19, 60, 36, 12, 59, 2, 57, 17, 42, 31, 1, 44, 16, 35, 47, 11, 32, 48, 13, 43, 1, 39, 51, 12, 57, 23, 6, 40, 53, 3, 55, 31, 39, 60, 35, 44, 5, 15, 45, 1, 62, 41, 26, 14, 47, 22, 36, 27, 50, 9, 26, 47, 52, 28, 54, 16, 1, 13, 51, 39, 23, 63, 1, 30, 15, 26, 2, 57, 19, 37, 1, 44, 21, 50, 13, 63, 8, 24, 56, 1, 35, 25, 58, 20, 2, 28, 14, 51, 33, 59, 13, 30, 4, 49, 31, 24, 63, 26, 33, 3, 58, 38, 62, 24, 32, 8, 17, 45, 5, 48, 18, 3, 43, 11 },
1068   { 21, 4, 24, 34, 59, 1, 37, 11, 53, 5, 47, 2, 22, 40, 32, 1, 24, 50, 21, 29, 38, 25, 63, 8, 55, 24, 53, 6, 62, 23, 59, 3, 54, 20, 58, 24, 5, 46, 15, 38, 48, 14, 27, 42, 23, 7, 46, 10, 17, 58, 25, 52, 23, 32, 49, 12, 55, 30, 40, 7, 59, 1, 56, 21, 39, 4, 23, 15, 37, 46, 55, 42, 21, 4, 48, 8, 45, 54, 37, 55, 32, 8, 46, 10, 30, 54, 4, 41, 25, 29, 36, 48, 11, 43, 14, 47, 5, 43, 53, 36, 61, 10, 45, 6, 41, 54, 27, 43, 16, 55, 6, 46, 18, 42, 23, 15, 1, 45, 12, 60, 37, 22, 62, 12, 39, 59, 16, 52 },
1069   { 47, 35, 56, 7, 19, 46, 31, 50, 33, 24, 61, 35, 50, 7, 53, 44, 55, 6, 46, 10, 52, 5, 21, 43, 36, 10, 18, 41, 26, 37, 8, 29, 40, 36, 9, 49, 34, 26, 61, 21, 7, 59, 18, 62, 29, 54, 20, 32, 51, 0, 40, 10, 55, 6, 20, 36, 9, 61, 5, 51, 44, 19, 33, 43, 13, 57, 40, 63, 8, 24, 29, 10, 60, 34, 27, 40, 25, 18, 10, 42, 21, 49, 26, 62, 38, 12, 33, 61, 5, 57, 2, 19, 54, 28, 62, 22, 38, 31, 16, 7, 22, 47, 29, 17, 35, 8, 20, 51, 2, 40, 22, 50, 13, 61, 28, 53, 35, 20, 56, 30, 2, 53, 14, 41, 23, 34, 8, 31 },
1070   { 12, 2, 42, 29, 52, 13, 21, 8, 55, 14, 41, 17, 28, 58, 23, 11, 17, 36, 31, 62, 17, 34, 50, 14, 28, 61, 33, 52, 2, 51, 17, 45, 7, 25, 62, 30, 18, 55, 0, 42, 30, 35, 45, 1, 12, 48, 3, 63, 21, 36, 30, 48, 19, 59, 43, 27, 46, 17, 34, 25, 12, 29, 53, 6, 48, 31, 11, 34, 49, 3, 36, 50, 19, 47, 14, 61, 11, 36, 58, 4, 60, 14, 39, 22, 6, 52, 15, 35, 17, 46, 31, 42, 9, 34, 3, 52, 12, 60, 26, 56, 40, 2, 53, 23, 57, 38, 62, 14, 36, 59, 10, 31, 39, 6, 49, 9, 41, 26, 5, 48, 43, 27, 33, 58, 1, 50, 25, 57 },
1071   { 61, 37, 15, 61, 3, 39, 58, 43, 26, 0, 44, 10, 47, 3, 37, 63, 28, 43, 13, 39, 3, 57, 30, 59, 0, 48, 5, 43, 13, 22, 60, 33, 55, 15, 42, 4, 52, 10, 45, 13, 54, 4, 24, 49, 37, 26, 41, 14, 42, 9, 61, 13, 38, 23, 3, 53, 0, 58, 21, 42, 63, 10, 17, 61, 25, 0, 58, 28, 17, 44, 57, 12, 27, 0, 55, 5, 52, 28, 23, 47, 29, 0, 43, 17, 58, 28, 47, 23, 55, 10, 58, 23, 51, 40, 18, 33, 45, 0, 49, 8, 32, 61, 19, 48, 0, 26, 7, 47, 29, 18, 44, 0, 56, 34, 20, 59, 15, 51, 37, 18, 10, 52, 7, 20, 46, 9, 38, 17 },
1072   { 6, 27, 48, 23, 45, 29, 5, 18, 38, 62, 27, 56, 20, 32, 15, 9, 48, 0, 54, 22, 45, 20, 7, 41, 23, 39, 19, 27, 58, 31, 44, 0, 12, 50, 23, 56, 20, 39, 32, 59, 16, 52, 33, 9, 57, 22, 6, 58, 28, 50, 24, 2, 56, 35, 16, 45, 32, 38, 15, 54, 2, 38, 46, 22, 35, 45, 20, 5, 52, 25, 7, 35, 59, 32, 22, 43, 38, 3, 51, 16, 34, 53, 32, 50, 3, 40, 8, 43, 0, 39, 27, 4, 14, 61, 8, 55, 15, 41, 20, 44, 27, 13, 39, 11, 46, 42, 54, 33, 4, 52, 23, 61, 14, 25, 43, 2, 33, 11, 63, 29, 61, 17, 40, 55, 22, 62, 28, 44 },
1073   { 20, 54, 8, 56, 35, 10, 63, 31, 52, 12, 48, 6, 59, 41, 52, 33, 19, 58, 25, 49, 11, 37, 47, 12, 54, 15, 56, 35, 7, 47, 16, 53, 28, 34, 5, 37, 28, 8, 48, 3, 28, 38, 18, 61, 16, 43, 53, 32, 4, 17, 47, 27, 44, 8, 63, 10, 25, 49, 6, 37, 24, 52, 32, 3, 50, 12, 41, 56, 38, 14, 62, 20, 40, 16, 53, 31, 18, 63, 41, 9, 59, 7, 13, 25, 57, 20, 63, 26, 53, 18, 48, 62, 30, 46, 21, 25, 58, 29, 36, 4, 55, 34, 6, 60, 31, 16, 21, 12, 58, 38, 9, 29, 47, 7, 52, 30, 57, 44, 22, 0, 35, 45, 3, 31, 14, 36, 0, 51 },
1074   { 42, 14, 33, 24, 16, 49, 40, 2, 22, 33, 16, 36, 25, 1, 21, 61, 38, 8, 33, 4, 62, 26, 29, 60, 6, 46, 30, 11, 63, 4, 36, 40, 19, 57, 46, 11, 41, 63, 22, 25, 58, 10, 46, 2, 34, 27, 11, 38, 56, 34, 12, 53, 18, 33, 41, 51, 13, 28, 60, 20, 47, 14, 29, 59, 16, 62, 8, 22, 32, 47, 9, 49, 2, 44, 7, 12, 45, 6, 20, 27, 45, 24, 62, 42, 36, 11, 33, 15, 37, 7, 32, 10, 37, 1, 35, 50, 6, 11, 63, 24, 52, 15, 50, 24, 3, 37, 56, 27, 34, 22, 49, 16, 36, 62, 17, 39, 4, 15, 54, 24, 50, 8, 58, 26, 49, 54, 11, 30 },
1075   { 4, 59, 41, 1, 53, 12, 25, 45, 59, 7, 51, 39, 54, 14, 46, 4, 27, 53, 16, 44, 18, 51, 1, 32, 25, 2, 50, 40, 20, 54, 24, 9, 62, 2, 27, 60, 1, 17, 36, 50, 6, 40, 30, 55, 41, 19, 49, 1, 21, 60, 40, 5, 62, 1, 22, 30, 57, 4, 43, 31, 1, 55, 40, 7, 27, 37, 30, 54, 1, 19, 42, 30, 56, 26, 62, 49, 24, 57, 37, 56, 2, 39, 16, 5, 30, 55, 3, 49, 60, 23, 56, 44, 17, 52, 13, 42, 28, 48, 18, 45, 9, 37, 21, 41, 58, 10, 48, 1, 63, 5, 41, 57, 2, 24, 12, 48, 27, 42, 32, 46, 13, 38, 19, 34, 5, 41, 25, 60 },
1076   { 39, 28, 21, 46, 32, 57, 36, 9, 19, 42, 4, 29, 11, 43, 30, 49, 13, 42, 35, 56, 9, 39, 15, 52, 36, 61, 18, 26, 45, 14, 31, 48, 21, 43, 14, 33, 49, 54, 14, 44, 21, 62, 13, 23, 8, 62, 15, 51, 44, 7, 30, 37, 20, 42, 56, 7, 39, 18, 50, 11, 61, 9, 19, 43, 57, 2, 48, 11, 39, 60, 28, 4, 37, 17, 35, 1, 33, 11, 31, 14, 48, 19, 35, 51, 46, 21, 44, 29, 12, 41, 2, 22, 58, 26, 54, 4, 59, 38, 2, 33, 57, 1, 63, 13, 28, 51, 15, 40, 18, 45, 8, 30, 43, 37, 54, 19, 8, 59, 21, 6, 60, 29, 55, 10, 63, 15, 47, 17 },
1077   { 3, 50, 10, 62, 18, 5, 27, 49, 60, 23, 55, 18, 62, 24, 56, 10, 59, 28, 2, 23, 34, 59, 43, 20, 10, 42, 8, 49, 1, 37, 57, 6, 51, 29, 53, 7, 23, 31, 5, 32, 51, 0, 35, 54, 45, 31, 5, 26, 36, 24, 55, 15, 48, 29, 14, 48, 26, 60, 21, 41, 36, 26, 50, 33, 14, 44, 17, 24, 52, 15, 46, 23, 54, 6, 47, 21, 60, 50, 4, 53, 29, 61, 8, 23, 1, 60, 19, 6, 53, 16, 47, 34, 6, 39, 16, 31, 12, 20, 53, 22, 30, 43, 25, 46, 35, 6, 44, 32, 53, 26, 55, 19, 11, 59, 5, 33, 51, 1, 35, 53, 25, 3, 42, 23, 44, 32, 7, 53 },
1078   { 22, 44, 37, 6, 26, 51, 38, 0, 34, 13, 31, 46, 3, 37, 6, 19, 40, 21, 47, 63, 12, 5, 29, 55, 22, 58, 34, 28, 60, 22, 11, 41, 17, 38, 9, 44, 59, 39, 56, 19, 11, 47, 25, 15, 3, 39, 57, 17, 61, 11, 46, 3, 58, 9, 54, 35, 2, 34, 8, 45, 15, 56, 5, 23, 53, 33, 63, 35, 4, 59, 10, 51, 13, 61, 29, 41, 15, 25, 43, 19, 40, 10, 54, 33, 41, 12, 38, 51, 31, 26, 61, 9, 30, 45, 24, 62, 49, 40, 10, 61, 14, 49, 5, 17, 54, 20, 60, 23, 3, 13, 35, 50, 32, 23, 46, 27, 38, 63, 16, 12, 39, 48, 18, 51, 1, 27, 56, 35 },
1079   { 63, 15, 30, 55, 43, 14, 57, 17, 53, 44, 7, 48, 26, 50, 32, 60, 0, 53, 14, 31, 50, 24, 46, 0, 38, 13, 4, 52, 16, 45, 30, 59, 0, 25, 55, 35, 16, 10, 26, 42, 58, 29, 60, 38, 50, 22, 28, 47, 0, 50, 28, 19, 33, 39, 11, 44, 16, 52, 24, 59, 3, 38, 27, 51, 0, 21, 7, 42, 26, 34, 21, 40, 33, 18, 39, 3, 54, 38, 8, 59, 0, 44, 27, 15, 58, 28, 57, 9, 43, 0, 36, 50, 20, 59, 8, 34, 0, 27, 47, 7, 36, 19, 56, 32, 0, 38, 11, 29, 62, 47, 6, 61, 0, 41, 14, 56, 10, 23, 45, 31, 57, 8, 36, 13, 58, 38, 11, 19 },
1080   { 0, 34, 12, 47, 21, 2, 40, 30, 11, 25, 61, 20, 40, 15, 35, 22, 45, 36, 7, 41, 17, 57, 9, 48, 32, 62, 44, 24, 35, 3, 54, 13, 33, 63, 19, 4, 48, 22, 62, 2, 37, 8, 33, 6, 20, 52, 9, 32, 43, 13, 39, 63, 25, 4, 49, 23, 62, 32, 9, 30, 48, 18, 63, 12, 46, 29, 58, 13, 48, 8, 57, 31, 0, 51, 9, 58, 12, 22, 47, 29, 35, 22, 49, 5, 46, 4, 34, 20, 63, 24, 56, 11, 41, 3, 51, 19, 56, 35, 17, 58, 28, 42, 9, 45, 59, 26, 51, 42, 17, 36, 25, 15, 53, 21, 44, 3, 30, 55, 5, 50, 21, 28, 61, 32, 6, 49, 28, 46 },
1081   { 58, 42, 60, 4, 31, 59, 22, 63, 35, 38, 9, 54, 1, 57, 8, 51, 16, 58, 27, 53, 3, 38, 30, 15, 27, 6, 19, 56, 10, 50, 21, 36, 47, 5, 43, 28, 51, 32, 13, 46, 18, 54, 16, 43, 63, 12, 36, 59, 22, 34, 5, 52, 17, 59, 27, 41, 0, 19, 55, 37, 13, 43, 6, 34, 41, 10, 36, 55, 19, 44, 3, 16, 58, 27, 49, 25, 32, 62, 17, 55, 13, 63, 18, 52, 25, 37, 17, 48, 13, 32, 5, 46, 28, 37, 14, 43, 25, 5, 51, 39, 3, 52, 33, 22, 8, 40, 12, 4, 57, 9, 46, 39, 28, 58, 13, 62, 17, 42, 19, 36, 0, 47, 16, 43, 24, 21, 54, 13 },
1082   { 25, 9, 23, 50, 36, 8, 45, 14, 3, 51, 16, 28, 44, 12, 42, 29, 4, 26, 10, 47, 22, 61, 18, 54, 51, 39, 46, 13, 41, 26, 58, 7, 18, 39, 12, 57, 15, 1, 52, 27, 41, 23, 48, 1, 27, 45, 18, 2, 57, 26, 55, 8, 43, 31, 6, 58, 14, 51, 40, 5, 61, 31, 24, 54, 17, 60, 22, 1, 39, 30, 53, 45, 36, 13, 43, 5, 45, 2, 37, 6, 34, 42, 2, 39, 10, 62, 7, 54, 40, 18, 60, 15, 52, 21, 63, 8, 55, 46, 15, 30, 23, 13, 62, 16, 50, 24, 58, 31, 48, 21, 34, 2, 49, 7, 31, 37, 26, 48, 9, 61, 40, 11, 52, 2, 60, 40, 4, 37 },
1083   { 52, 28, 39, 16, 54, 19, 29, 55, 42, 20, 58, 33, 24, 63, 18, 55, 39, 62, 43, 34, 12, 40, 6, 35, 2, 25, 8, 62, 34, 1, 31, 42, 61, 27, 53, 24, 40, 61, 34, 8, 59, 4, 30, 56, 40, 6, 53, 42, 10, 48, 16, 37, 12, 46, 21, 36, 47, 11, 28, 45, 22, 10, 57, 2, 49, 31, 14, 44, 61, 11, 25, 6, 23, 63, 18, 36, 28, 56, 20, 51, 11, 48, 27, 56, 32, 22, 45, 30, 2, 42, 27, 39, 1, 44, 23, 31, 38, 22, 11, 61, 43, 54, 4, 47, 35, 2, 44, 16, 28, 54, 12, 62, 18, 43, 10, 52, 1, 58, 33, 15, 29, 56, 20, 34, 9, 30, 48, 17 },
1084   { 46, 2, 56, 11, 41, 1, 49, 6, 27, 47, 2, 48, 5, 32, 37, 3, 13, 19, 32, 1, 55, 28, 60, 17, 43, 59, 32, 20, 49, 16, 55, 23, 14, 46, 2, 36, 6, 30, 20, 49, 12, 47, 35, 14, 21, 60, 29, 14, 35, 24, 46, 1, 56, 29, 53, 8, 33, 23, 56, 1, 35, 46, 20, 39, 26, 4, 53, 28, 17, 38, 60, 34, 48, 9, 55, 15, 46, 7, 41, 31, 60, 24, 16, 36, 1, 59, 19, 52, 35, 6, 55, 11, 59, 33, 7, 57, 4, 29, 48, 1, 19, 26, 37, 30, 18, 63, 37, 6, 59, 1, 40, 24, 56, 33, 46, 22, 35, 7, 24, 53, 39, 5, 26, 45, 55, 18, 62, 7 },
1085   { 20, 60, 29, 34, 20, 62, 33, 52, 10, 36, 13, 60, 41, 21, 50, 27, 56, 49, 8, 51, 21, 45, 11, 48, 8, 23, 53, 3, 29, 44, 5, 52, 9, 32, 50, 17, 43, 56, 3, 38, 24, 10, 62, 25, 51, 9, 33, 49, 61, 7, 30, 62, 22, 19, 2, 42, 63, 5, 49, 18, 60, 15, 52, 7, 43, 56, 23, 50, 5, 50, 2, 20, 41, 30, 1, 52, 22, 61, 14, 26, 3, 43, 53, 7, 47, 28, 11, 14, 23, 58, 33, 25, 47, 13, 50, 17, 40, 54, 34, 60, 41, 6, 59, 14, 50, 7, 25, 55, 20, 42, 51, 8, 27, 4, 16, 60, 28, 50, 44, 3, 22, 49, 63, 12, 33, 1, 43, 31 },
1086   { 36, 5, 46, 8, 44, 24, 13, 39, 25, 57, 31, 18, 8, 52, 10, 45, 6, 30, 36, 24, 63, 4, 33, 26, 57, 40, 15, 56, 37, 12, 40, 25, 37, 58, 11, 63, 21, 45, 16, 60, 31, 53, 18, 33, 3, 45, 23, 0, 20, 54, 40, 15, 50, 38, 60, 16, 25, 42, 29, 38, 7, 41, 25, 62, 18, 33, 8, 35, 42, 16, 32, 56, 12, 39, 59, 19, 34, 9, 49, 38, 57, 12, 21, 50, 14, 40, 61, 44, 50, 9, 49, 19, 3, 29, 35, 62, 12, 24, 7, 18, 52, 32, 10, 46, 21, 41, 32, 11, 36, 29, 14, 34, 60, 38, 54, 11, 41, 14, 19, 57, 32, 16, 7, 41, 51, 25, 14, 57 },
1087   { 53, 18, 26, 50, 15, 58, 4, 63, 17, 43, 7, 40, 61, 35, 15, 41, 23, 60, 16, 38, 14, 42, 19, 50, 0, 31, 10, 46, 27, 63, 18, 60, 0, 20, 29, 39, 8, 26, 37, 5, 42, 0, 44, 39, 57, 17, 58, 41, 28, 37, 4, 32, 9, 44, 12, 31, 54, 10, 59, 14, 27, 53, 12, 36, 0, 47, 13, 63, 21, 58, 10, 24, 50, 27, 4, 26, 44, 53, 31, 0, 18, 42, 29, 33, 57, 4, 32, 26, 0, 38, 16, 61, 41, 53, 20, 0, 42, 44, 49, 27, 10, 56, 39, 0, 57, 15, 53, 49, 3, 61, 22, 47, 17, 5, 49, 26, 2, 63, 39, 10, 47, 27, 37, 23, 4, 59, 38, 10 },
1088   { 23, 39, 61, 3, 37, 28, 48, 31, 0, 34, 51, 23, 2, 26, 58, 0, 53, 11, 46, 1, 57, 29, 52, 14, 37, 61, 21, 35, 2, 49, 7, 34, 47, 55, 4, 33, 54, 13, 58, 52, 19, 50, 22, 7, 13, 29, 36, 11, 51, 17, 60, 25, 55, 4, 34, 51, 0, 35, 20, 48, 32, 3, 51, 30, 59, 28, 40, 3, 46, 29, 54, 43, 7, 62, 47, 11, 39, 4, 23, 46, 55, 8, 63, 5, 25, 37, 18, 46, 21, 56, 31, 5, 36, 8, 45, 58, 26, 15, 2, 36, 47, 21, 29, 44, 25, 34, 3, 27, 43, 10, 52, 0, 45, 30, 24, 36, 43, 18, 34, 59, 0, 52, 61, 15, 44, 19, 30, 49 },
1089   { 0, 27, 12, 43, 54, 9, 22, 53, 21, 46, 15, 55, 29, 47, 20, 33, 39, 28, 59, 35, 9, 44, 5, 24, 47, 7, 52, 17, 56, 22, 30, 42, 14, 26, 45, 18, 49, 1, 24, 34, 11, 27, 55, 32, 61, 47, 2, 56, 6, 44, 13, 47, 36, 27, 58, 22, 16, 47, 40, 4, 57, 38, 21, 45, 16, 9, 56, 26, 11, 38, 0, 22, 36, 17, 33, 57, 16, 30, 62, 15, 35, 40, 20, 45, 59, 10, 54, 8, 63, 13, 52, 27, 22, 57, 28, 12, 32, 51, 55, 22, 63, 4, 16, 54, 12, 62, 45, 19, 58, 13, 32, 40, 20, 56, 7, 57, 9, 54, 6, 29, 42, 21, 8, 55, 35, 47, 6, 41 },
1090   { 56, 33, 58, 32, 19, 35, 42, 6, 59, 11, 38, 5, 49, 12, 62, 7, 52, 17, 5, 25, 54, 20, 61, 31, 54, 27, 41, 11, 44, 5, 59, 12, 36, 51, 10, 61, 28, 41, 48, 9, 43, 63, 5, 40, 20, 8, 49, 26, 34, 21, 58, 1, 18, 45, 7, 39, 61, 26, 8, 50, 23, 10, 63, 5, 55, 37, 19, 49, 52, 15, 59, 47, 13, 54, 1, 25, 42, 58, 10, 48, 3, 27, 50, 1, 17, 48, 34, 41, 16, 40, 2, 45, 10, 39, 17, 61, 5, 38, 19, 9, 41, 31, 60, 38, 5, 23, 36, 8, 30, 55, 24, 63, 12, 48, 14, 51, 31, 20, 45, 25, 12, 50, 32, 2, 28, 11, 62, 14 },
1091   { 44, 16, 7, 48, 1, 62, 16, 50, 27, 33, 61, 25, 17, 44, 31, 14, 22, 43, 32, 48, 18, 40, 8, 36, 3, 16, 33, 62, 23, 38, 25, 53, 2, 21, 41, 6, 22, 15, 59, 29, 16, 37, 26, 15, 52, 42, 23, 15, 54, 39, 10, 30, 53, 11, 49, 24, 2, 43, 55, 17, 34, 44, 15, 31, 24, 44, 2, 32, 7, 35, 25, 5, 40, 45, 29, 51, 6, 21, 37, 52, 24, 60, 13, 31, 53, 23, 2, 28, 49, 24, 31, 60, 20, 51, 1, 34, 48, 14, 59, 33, 50, 1, 18, 33, 48, 60, 17, 51, 39, 6, 38, 2, 35, 29, 40, 23, 1, 62, 15, 53, 37, 17, 46, 57, 40, 51, 24, 22 },
1092   { 5, 37, 52, 24, 45, 13, 40, 3, 45, 9, 19, 42, 56, 4, 37, 46, 56, 2, 63, 11, 51, 1, 49, 13, 59, 45, 39, 1, 48, 15, 58, 9, 46, 31, 54, 35, 57, 38, 3, 46, 56, 4, 47, 57, 1, 30, 38, 63, 3, 46, 28, 63, 41, 14, 33, 62, 19, 32, 13, 28, 61, 1, 53, 42, 11, 60, 22, 62, 27, 42, 61, 31, 19, 8, 61, 12, 32, 55, 2, 18, 33, 12, 43, 36, 9, 62, 30, 55, 6, 58, 35, 7, 43, 29, 54, 23, 43, 30, 3, 25, 11, 45, 52, 28, 7, 14, 42, 1, 22, 50, 16, 53, 19, 59, 4, 46, 33, 41, 4, 35, 58, 5, 26, 13, 20, 2, 34, 54 },
1093   { 30, 63, 21, 10, 26, 55, 29, 59, 23, 39, 53, 1, 36, 24, 59, 27, 10, 34, 23, 38, 30, 60, 22, 42, 28, 19, 9, 57, 30, 19, 43, 33, 13, 63, 3, 19, 11, 50, 31, 20, 14, 34, 10, 35, 17, 59, 7, 31, 19, 25, 50, 5, 20, 57, 29, 6, 52, 41, 4, 46, 20, 37, 26, 17, 49, 6, 39, 18, 53, 14, 3, 49, 57, 23, 34, 48, 14, 41, 28, 38, 56, 6, 58, 25, 39, 19, 43, 15, 37, 11, 47, 18, 53, 4, 37, 9, 62, 21, 53, 40, 57, 24, 13, 40, 56, 26, 47, 31, 59, 25, 45, 27, 10, 43, 21, 61, 13, 27, 48, 9, 23, 43, 31, 62, 38, 59, 9, 47 },
1094   { 25, 4, 40, 60, 34, 6, 18, 36, 8, 57, 12, 30, 49, 14, 6, 54, 41, 16, 50, 6, 43, 15, 34, 4, 53, 24, 50, 35, 4, 51, 7, 55, 28, 24, 39, 44, 60, 7, 25, 62, 42, 53, 24, 61, 28, 45, 52, 12, 48, 37, 9, 35, 43, 3, 37, 48, 12, 58, 30, 52, 9, 59, 6, 57, 33, 29, 48, 4, 37, 45, 20, 34, 10, 39, 0, 60, 22, 45, 8, 63, 21, 42, 14, 49, 3, 56, 11, 46, 21, 61, 0, 42, 25, 13, 63, 17, 36, 8, 46, 16, 6, 35, 63, 0, 21, 37, 4, 57, 9, 34, 5, 61, 48, 32, 8, 37, 54, 17, 56, 30, 60, 0, 50, 16, 7, 29, 42, 17 },
1095   { 32, 50, 15, 48, 2, 43, 52, 25, 47, 16, 32, 63, 21, 52, 40, 19, 0, 61, 29, 58, 20, 56, 26, 46, 12, 55, 6, 22, 62, 32, 17, 40, 0, 49, 34, 8, 27, 32, 48, 0, 21, 39, 5, 44, 12, 6, 22, 40, 0, 57, 16, 60, 23, 17, 54, 22, 36, 15, 24, 39, 19, 34, 47, 23, 0, 54, 13, 51, 24, 9, 55, 16, 52, 27, 44, 20, 4, 54, 26, 49, 0, 30, 46, 16, 29, 51, 34, 4, 52, 28, 33, 15, 57, 39, 26, 49, 0, 56, 27, 31, 48, 20, 43, 29, 53, 11, 46, 19, 41, 13, 55, 18, 0, 57, 26, 51, 2, 44, 6, 38, 14, 40, 22, 45, 36, 53, 3, 57 },
1096   { 44, 12, 37, 28, 22, 57, 11, 38, 0, 51, 9, 41, 4, 29, 11, 47, 33, 45, 12, 26, 3, 36, 9, 63, 31, 16, 38, 44, 14, 47, 25, 61, 20, 58, 15, 47, 17, 57, 13, 36, 9, 51, 18, 29, 50, 36, 54, 20, 61, 27, 32, 13, 53, 44, 9, 27, 0, 63, 45, 2, 56, 10, 14, 43, 41, 28, 58, 11, 35, 60, 30, 41, 6, 63, 11, 51, 37, 32, 15, 10, 35, 53, 5, 61, 22, 7, 26, 59, 23, 9, 44, 48, 21, 3, 51, 32, 24, 41, 12, 61, 2, 55, 9, 15, 35, 58, 28, 15, 62, 30, 37, 23, 42, 29, 11, 17, 35, 24, 63, 20, 52, 28, 8, 55, 11, 23, 47, 19 },
1097   { 0, 56, 8, 53, 14, 31, 61, 20, 55, 28, 62, 18, 35, 60, 25, 57, 7, 23, 39, 54, 47, 17, 43, 0, 40, 59, 29, 2, 56, 10, 37, 5, 43, 11, 29, 52, 1, 23, 54, 41, 59, 30, 55, 1, 62, 15, 33, 4, 43, 10, 47, 39, 1, 31, 40, 60, 49, 33, 7, 55, 26, 50, 31, 61, 8, 18, 21, 32, 44, 1, 25, 47, 18, 36, 30, 23, 59, 7, 40, 59, 27, 19, 38, 32, 44, 54, 40, 17, 38, 60, 27, 6, 35, 55, 10, 14, 44, 5, 50, 17, 38, 26, 42, 50, 18, 3, 44, 52, 2, 49, 7, 52, 15, 46, 62, 39, 55, 10, 31, 48, 3, 58, 33, 18, 61, 34, 13, 59 },
1098   { 39, 27, 63, 20, 35, 41, 4, 45, 26, 5, 38, 13, 44, 2, 50, 17, 37, 52, 2, 13, 28, 58, 24, 51, 21, 8, 34, 48, 27, 42, 18, 51, 31, 56, 5, 36, 38, 44, 4, 17, 26, 11, 38, 23, 42, 8, 56, 39, 24, 51, 5, 56, 21, 59, 14, 6, 18, 42, 22, 35, 16, 37, 3, 25, 39, 46, 63, 5, 50, 17, 58, 8, 55, 3, 50, 12, 43, 17, 47, 2, 51, 9, 62, 12, 1, 35, 13, 50, 1, 37, 12, 51, 19, 29, 46, 59, 22, 58, 33, 45, 22, 60, 10, 32, 61, 39, 8, 33, 25, 36, 20, 60, 38, 4, 21, 5, 28, 45, 12, 18, 42, 11, 49, 1, 27, 40, 6, 30 },
1099   { 24, 16, 42, 1, 50, 10, 48, 17, 33, 43, 24, 48, 21, 55, 31, 42, 10, 21, 63, 35, 49, 6, 33, 13, 41, 53, 10, 20, 60, 6, 53, 26, 12, 41, 22, 60, 14, 28, 63, 33, 49, 3, 45, 16, 48, 26, 14, 46, 18, 30, 35, 26, 8, 50, 29, 51, 25, 57, 12, 47, 53, 9, 62, 20, 54, 2, 36, 15, 40, 28, 33, 13, 38, 24, 46, 1, 29, 56, 33, 20, 44, 24, 41, 26, 57, 20, 63, 8, 30, 55, 5, 41, 62, 8, 34, 2, 37, 10, 19, 6, 37, 1, 53, 23, 5, 27, 58, 22, 43, 12, 50, 26, 9, 34, 54, 32, 49, 1, 59, 37, 22, 46, 25, 36, 51, 15, 54, 46 },
1100   { 52, 7, 45, 33, 26, 58, 14, 60, 7, 54, 3, 58, 8, 34, 14, 5, 59, 30, 18, 44, 8, 22, 48, 62, 3, 26, 55, 38, 23, 16, 39, 1, 62, 24, 49, 9, 53, 19, 46, 7, 19, 60, 31, 58, 2, 34, 53, 7, 59, 2, 62, 42, 46, 19, 36, 11, 44, 4, 38, 28, 1, 43, 32, 51, 12, 29, 56, 22, 52, 2, 62, 49, 22, 60, 14, 35, 63, 5, 25, 57, 14, 53, 4, 46, 18, 31, 42, 22, 47, 20, 58, 31, 16, 43, 23, 54, 30, 42, 52, 57, 29, 49, 30, 13, 45, 48, 16, 55, 6, 63, 1, 44, 14, 58, 19, 47, 15, 24, 51, 34, 6, 55, 5, 63, 20, 41, 21, 9 },
1101   { 30, 62, 18, 55, 5, 23, 39, 29, 49, 30, 15, 36, 28, 46, 60, 25, 39, 46, 4, 32, 61, 40, 15, 30, 36, 45, 14, 2, 49, 33, 57, 45, 18, 32, 3, 45, 30, 2, 35, 52, 40, 27, 13, 21, 38, 63, 20, 28, 37, 23, 16, 10, 13, 55, 2, 62, 21, 32, 60, 17, 58, 23, 5, 40, 16, 48, 7, 45, 10, 26, 43, 19, 6, 31, 52, 21, 39, 16, 48, 9, 37, 28, 36, 55, 7, 48, 3, 59, 15, 45, 25, 1, 53, 13, 47, 7, 62, 15, 4, 25, 12, 41, 18, 60, 38, 11, 34, 19, 39, 31, 29, 56, 23, 42, 3, 27, 60, 41, 8, 16, 61, 29, 43, 9, 32, 2, 60, 34 },
1102   { 3, 38, 13, 37, 52, 44, 2, 19, 12, 42, 63, 19, 40, 1, 20, 50, 12, 55, 15, 56, 27, 1, 54, 11, 57, 18, 32, 63, 44, 4, 29, 13, 37, 61, 35, 16, 42, 57, 12, 22, 6, 55, 43, 10, 50, 5, 44, 11, 48, 52, 34, 58, 28, 41, 38, 30, 7, 52, 11, 49, 30, 14, 45, 27, 59, 34, 21, 38, 32, 58, 11, 36, 56, 42, 9, 41, 3, 54, 31, 42, 0, 60, 16, 11, 39, 24, 52, 33, 6, 36, 10, 40, 32, 60, 26, 20, 39, 28, 47, 34, 63, 8, 54, 3, 24, 56, 0, 51, 13, 47, 16, 40, 7, 35, 52, 11, 36, 4, 57, 30, 39, 13, 18, 50, 58, 28, 12, 48 },
1103   { 57, 24, 49, 21, 10, 31, 61, 36, 56, 0, 22, 53, 11, 56, 32, 7, 36, 27, 41, 9, 46, 19, 34, 42, 25, 7, 50, 9, 28, 21, 54, 8, 50, 7, 27, 59, 10, 25, 48, 62, 37, 0, 33, 58, 25, 18, 32, 61, 0, 15, 45, 5, 50, 3, 23, 55, 47, 17, 40, 6, 60, 34, 53, 8, 41, 0, 61, 13, 54, 4, 46, 28, 0, 17, 48, 27, 58, 13, 23, 61, 33, 21, 50, 30, 62, 8, 14, 29, 56, 27, 61, 49, 17, 2, 44, 11, 51, 0, 59, 17, 40, 20, 32, 47, 36, 21, 42, 28, 60, 4, 54, 10, 59, 17, 30, 62, 21, 43, 26, 48, 0, 56, 36, 25, 8, 44, 39, 17 },
1104   { 10, 42, 4, 59, 27, 47, 8, 23, 51, 32, 45, 6, 37, 26, 48, 43, 62, 0, 21, 53, 38, 12, 51, 5, 60, 47, 24, 37, 59, 15, 35, 47, 22, 55, 0, 50, 21, 40, 6, 29, 15, 52, 24, 8, 41, 55, 13, 29, 40, 56, 24, 31, 19, 33, 61, 15, 0, 35, 24, 42, 21, 2, 19, 57, 24, 15, 30, 50, 20, 25, 40, 16, 57, 34, 61, 8, 29, 45, 6, 49, 11, 47, 2, 44, 19, 57, 38, 50, 12, 42, 21, 4, 35, 52, 28, 56, 23, 36, 13, 45, 4, 52, 27, 14, 6, 62, 9, 45, 21, 37, 25, 46, 33, 49, 0, 44, 7, 53, 13, 19, 53, 31, 3, 47, 15, 56, 22, 51 },
1105   { 35, 28, 53, 32, 1, 16, 54, 40, 9, 17, 25, 58, 14, 59, 3, 22, 16, 51, 31, 5, 23, 58, 28, 17, 35, 20, 0, 42, 11, 52, 3, 31, 41, 17, 43, 13, 32, 54, 18, 60, 32, 45, 17, 49, 2, 36, 51, 22, 7, 36, 9, 63, 48, 12, 46, 26, 43, 28, 63, 13, 48, 37, 51, 33, 5, 47, 55, 9, 42, 63, 7, 51, 24, 12, 37, 19, 55, 34, 18, 38, 15, 28, 54, 34, 5, 43, 22, 0, 48, 14, 54, 24, 58, 9, 38, 5, 32, 55, 21, 30, 49, 9, 59, 43, 30, 51, 35, 26, 7, 53, 2, 22, 14, 27, 57, 18, 38, 24, 33, 45, 10, 41, 20, 60, 37, 5, 32, 0 },
1106   { 63, 19, 15, 40, 62, 35, 14, 28, 46, 61, 4, 49, 35, 10, 29, 54, 33, 8, 45, 62, 37, 1, 43, 55, 10, 52, 61, 30, 19, 40, 25, 62, 11, 38, 27, 58, 36, 3, 46, 8, 39, 4, 62, 28, 47, 20, 4, 54, 47, 27, 43, 1, 21, 38, 8, 58, 10, 54, 4, 56, 9, 26, 12, 39, 60, 27, 18, 37, 1, 31, 35, 5, 45, 50, 2, 43, 26, 1, 59, 23, 56, 40, 7, 26, 58, 17, 32, 63, 25, 39, 7, 31, 45, 19, 63, 15, 48, 8, 37, 61, 16, 34, 1, 56, 18, 3, 15, 58, 49, 32, 63, 41, 55, 5, 40, 22, 50, 6, 59, 2, 63, 23, 52, 11, 26, 61, 44, 23 },
1107   { 11, 56, 46, 6, 22, 43, 58, 3, 34, 21, 38, 30, 18, 44, 52, 13, 41, 57, 17, 28, 14, 49, 25, 7, 33, 39, 26, 6, 56, 48, 1, 20, 56, 5, 46, 9, 19, 51, 30, 25, 56, 21, 35, 14, 57, 42, 16, 33, 10, 57, 17, 59, 41, 25, 53, 37, 20, 40, 30, 18, 31, 62, 44, 22, 3, 44, 11, 48, 23, 53, 18, 60, 29, 22, 62, 15, 53, 47, 10, 41, 3, 19, 52, 36, 13, 46, 10, 35, 3, 61, 41, 16, 1, 50, 26, 42, 18, 46, 2, 25, 54, 20, 39, 23, 47, 31, 41, 12, 38, 17, 8, 19, 31, 48, 12, 61, 9, 54, 29, 35, 15, 38, 6, 43, 34, 14, 7, 47 },
1108   { 39, 2, 33, 26, 53, 8, 18, 50, 41, 12, 53, 1, 63, 24, 19, 39, 2, 24, 47, 10, 60, 38, 19, 63, 48, 4, 15, 45, 32, 14, 60, 36, 29, 53, 23, 63, 34, 12, 61, 1, 43, 11, 53, 30, 1, 26, 60, 45, 23, 39, 3, 29, 12, 50, 4, 16, 51, 3, 45, 36, 50, 1, 16, 54, 35, 14, 57, 30, 58, 9, 46, 14, 41, 10, 32, 38, 4, 30, 21, 51, 32, 63, 25, 1, 60, 27, 53, 18, 51, 22, 28, 55, 34, 12, 40, 3, 60, 29, 57, 41, 6, 44, 11, 53, 8, 61, 24, 57, 1, 28, 44, 59, 36, 3, 34, 25, 41, 31, 16, 44, 22, 47, 28, 58, 1, 49, 54, 29 },
1109   { 58, 25, 50, 13, 38, 30, 60, 24, 6, 57, 27, 42, 9, 45, 6, 61, 30, 50, 4, 34, 29, 3, 46, 13, 22, 42, 58, 28, 9, 39, 23, 44, 7, 15, 44, 2, 40, 15, 47, 41, 23, 37, 7, 59, 38, 11, 34, 6, 62, 14, 52, 35, 55, 19, 32, 61, 33, 24, 57, 6, 22, 59, 29, 7, 49, 25, 40, 3, 17, 39, 27, 52, 0, 55, 16, 57, 24, 61, 36, 6, 29, 12, 48, 39, 20, 44, 6, 40, 33, 5, 48, 10, 57, 36, 22, 51, 33, 9, 24, 12, 62, 29, 50, 35, 14, 43, 5, 33, 47, 52, 13, 23, 10, 51, 56, 16, 46, 1, 49, 4, 61, 9, 52, 18, 31, 21, 36, 17 },
1110   { 19, 42, 9, 48, 2, 44, 11, 37, 48, 20, 33, 16, 55, 35, 49, 15, 37, 20, 59, 16, 53, 22, 56, 31, 50, 11, 34, 54, 16, 51, 4, 49, 33, 53, 21, 28, 56, 24, 31, 9, 52, 16, 48, 24, 44, 13, 51, 20, 31, 49, 18, 6, 34, 2, 44, 14, 47, 8, 15, 43, 13, 41, 33, 52, 20, 61, 7, 51, 34, 62, 4, 20, 36, 33, 43, 8, 46, 13, 53, 17, 45, 42, 9, 31, 52, 11, 30, 56, 13, 59, 17, 44, 27, 6, 62, 11, 43, 17, 49, 38, 26, 2, 16, 27, 58, 21, 54, 18, 26, 5, 35, 61, 43, 27, 7, 39, 14, 58, 37, 55, 20, 33, 13, 40, 62, 10, 55, 5 },
1111   { 51, 14, 61, 29, 59, 20, 55, 31, 0, 49, 11, 60, 3, 26, 22, 56, 0, 40, 12, 43, 41, 8, 36, 0, 17, 57, 24, 2, 46, 26, 61, 18, 0, 38, 12, 59, 6, 49, 3, 57, 19, 63, 5, 33, 18, 54, 28, 56, 0, 43, 26, 46, 63, 27, 56, 22, 27, 54, 38, 28, 63, 24, 10, 45, 0, 31, 42, 21, 12, 25, 44, 49, 59, 6, 26, 50, 3, 34, 27, 59, 0, 35, 62, 16, 4, 58, 47, 0, 43, 24, 37, 2, 54, 20, 46, 31, 0, 56, 34, 5, 55, 45, 60, 37, 0, 40, 10, 38, 63, 46, 15, 20, 0, 53, 21, 62, 30, 11, 24, 27, 40, 0, 57, 26, 3, 45, 27, 35 },
1112 };
1113
1114 #else
1115 #define DM_WIDTH 8
1116 #define DM_WIDTH_SHIFT 3
1117 #define DM_HEIGHT 8
1118 static const guchar DM[8][8] =
1119 {
1120   { 0,  32, 8,  40, 2,  34, 10, 42 },
1121   { 48, 16, 56, 24, 50, 18, 58, 26 },
1122   { 12, 44, 4,  36, 14, 46, 6,  38 },
1123   { 60, 28, 52, 20, 62, 30, 54, 22 },
1124   { 3,  35, 11, 43, 1,  33, 9,  41 },
1125   { 51, 19, 59, 27, 49, 17, 57, 25 },
1126   { 15, 47, 7,  39, 13, 45, 5,  37 },
1127   { 63, 31, 55, 23, 61, 29, 53, 21 }
1128 };
1129 #endif
1130
1131 static guint32 *DM_565 = NULL;
1132
1133 static void
1134 gdk_rgb_preprocess_dm_565 (void)
1135 {
1136   int i;
1137   guint32 dith;
1138
1139   if (DM_565 == NULL)
1140     {
1141       DM_565 = g_new (guint32, DM_WIDTH * DM_HEIGHT);
1142       for (i = 0; i < DM_WIDTH * DM_HEIGHT; i++)
1143         {
1144           dith = DM[0][i] >> 3;
1145           DM_565[i] = (dith << 20) | dith | (((7 - dith) >> 1) << 10);
1146 #ifdef VERBOSE
1147           g_print ("%i %x %x\n", i, dith, DM_565[i]);
1148 #endif
1149         }
1150     }
1151 }
1152
1153 static void
1154 gdk_rgb_convert_8_d666 (GdkRgbInfo *image_info, GdkImage *image,
1155                         gint x0, gint y0, gint width, gint height,
1156                         guchar *buf, int rowstride,
1157                         gint x_align, gint y_align, GdkRgbCmap *cmap)
1158 {
1159   int x, y;
1160   gint bpl;
1161   guchar *obuf, *obptr;
1162   guchar *bptr, *bp2;
1163   gint r, g, b;
1164   const guchar *dmp;
1165   gint dith;
1166   guchar *colorcube_d = image_info->colorcube_d;
1167
1168   bptr = buf;
1169   bpl = image->bpl;
1170   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
1171   for (y = 0; y < height; y++)
1172     {
1173       dmp = DM[(y_align + y) & (DM_HEIGHT - 1)];
1174       bp2 = bptr;
1175       obptr = obuf;
1176       for (x = 0; x < width; x++)
1177         {
1178           r = *bp2++;
1179           g = *bp2++;
1180           b = *bp2++;
1181           dith = (dmp[(x_align + x) & (DM_WIDTH - 1)] << 2) | 7;
1182           r = ((r * 5) + dith) >> 8;
1183           g = ((g * 5) + (262 - dith)) >> 8;
1184           b = ((b * 5) + dith) >> 8;
1185           obptr[0] = colorcube_d[(r << 6) | (g << 3) | b];
1186           obptr++;
1187         }
1188       bptr += rowstride;
1189       obuf += bpl;
1190     }
1191 }
1192
1193 static void
1194 gdk_rgb_convert_8_d (GdkRgbInfo *image_info, GdkImage *image,
1195                      gint x0, gint y0, gint width, gint height,
1196                      guchar *buf, int rowstride,
1197                      gint x_align, gint y_align,
1198                      GdkRgbCmap *cmap)
1199 {
1200   int x, y;
1201   gint bpl;
1202   guchar *obuf, *obptr;
1203   guchar *bptr, *bp2;
1204   gint r, g, b;
1205   const guchar *dmp;
1206   gint dith;
1207   gint rs, gs, bs;
1208   guchar *colorcube_d = image_info->colorcube_d;
1209
1210   bptr = buf;
1211   bpl = image->bpl;
1212   rs = image_info->nred_shades - 1;
1213   gs = image_info->ngreen_shades - 1;
1214   bs = image_info->nblue_shades - 1;
1215   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
1216   for (y = 0; y < height; y++)
1217     {
1218       dmp = DM[(y_align + y) & (DM_HEIGHT - 1)];
1219       bp2 = bptr;
1220       obptr = obuf;
1221       for (x = 0; x < width; x++)
1222         {
1223           r = *bp2++;
1224           g = *bp2++;
1225           b = *bp2++;
1226           dith = (dmp[(x_align + x) & (DM_WIDTH - 1)] << 2) | 7;
1227           r = ((r * rs) + dith) >> 8;
1228           g = ((g * gs) + (262 - dith)) >> 8;
1229           b = ((b * bs) + dith) >> 8;
1230           obptr[0] = colorcube_d[(r << 6) | (g << 3) | b];
1231           obptr++;
1232         }
1233       bptr += rowstride;
1234       obuf += bpl;
1235     }
1236 }
1237
1238 static void
1239 gdk_rgb_convert_8_indexed (GdkRgbInfo *image_info, GdkImage *image,
1240                            gint x0, gint y0, gint width, gint height,
1241                            guchar *buf, int rowstride,
1242                            gint x_align, gint y_align, GdkRgbCmap *cmap)
1243 {
1244   int x, y;
1245   gint bpl;
1246   guchar *obuf, *obptr;
1247   guchar *bptr, *bp2;
1248   guchar c;
1249   guchar *lut;
1250   GdkRgbCmapInfo *cmap_info = gdk_rgb_cmap_get_info (cmap, image_info);
1251
1252   lut = cmap_info->lut;
1253   bptr = buf;
1254   bpl = image->bpl;
1255   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
1256   for (y = 0; y < height; y++)
1257     {
1258       bp2 = bptr;
1259       obptr = obuf;
1260       for (x = 0; x < width; x++)
1261         {
1262           c = *bp2++;
1263           obptr[0] = lut[c];
1264           obptr++;
1265         }
1266       bptr += rowstride;
1267       obuf += bpl;
1268     }
1269 }
1270
1271 static void
1272 gdk_rgb_convert_gray8 (GdkRgbInfo *image_info, GdkImage *image,
1273                        gint x0, gint y0, gint width, gint height,
1274                        guchar *buf, int rowstride,
1275                        gint x_align, gint y_align, GdkRgbCmap *cmap)
1276 {
1277   int x, y;
1278   gint bpl;
1279   guchar *obuf, *obptr;
1280   guchar *bptr, *bp2;
1281   gint r, g, b;
1282
1283   bptr = buf;
1284   bpl = image->bpl;
1285   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
1286   for (y = 0; y < height; y++)
1287     {
1288       bp2 = bptr;
1289       obptr = obuf;
1290       for (x = 0; x < width; x++)
1291         {
1292           r = *bp2++;
1293           g = *bp2++;
1294           b = *bp2++;
1295           obptr[0] = (g + ((b + r) >> 1)) >> 1;
1296           obptr++;
1297         }
1298       bptr += rowstride;
1299       obuf += bpl;
1300     }
1301 }
1302
1303 static void
1304 gdk_rgb_convert_gray8_gray (GdkRgbInfo *image_info, GdkImage *image,
1305                             gint x0, gint y0, gint width, gint height,
1306                             guchar *buf, int rowstride,
1307                             gint x_align, gint y_align, GdkRgbCmap *cmap)
1308 {
1309   int y;
1310   gint bpl;
1311   guchar *obuf;
1312   guchar *bptr;
1313
1314   bptr = buf;
1315   bpl = image->bpl;
1316   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
1317   for (y = 0; y < height; y++)
1318     {
1319       memcpy (obuf, bptr, width);
1320       bptr += rowstride;
1321       obuf += bpl;
1322     }
1323 }
1324
1325 #if G_BYTE_ORDER == G_LITTLE_ENDIAN
1326 #define HAIRY_CONVERT_565
1327 #endif
1328
1329 #ifdef HAIRY_CONVERT_565
1330 /* Render a 24-bit RGB image in buf into the GdkImage, without dithering.
1331    This assumes native byte ordering - what should really be done is to
1332    check whether the the image byte_order is consistent with the _ENDIAN
1333    config flag, and if not, use a different function.
1334
1335    This one is even faster than the one below - its inner loop loads 3
1336    words (i.e. 4 24-bit pixels), does a lot of shifting and masking,
1337    then writes 2 words. */
1338 static void
1339 gdk_rgb_convert_565 (GdkRgbInfo *image_info, GdkImage *image,
1340                      gint x0, gint y0, gint width, gint height,
1341                      guchar *buf, int rowstride,
1342                      gint x_align, gint y_align, GdkRgbCmap *cmap)
1343 {
1344   int x, y;
1345   guchar *obuf, *obptr;
1346   gint bpl;
1347   guchar *bptr, *bp2;
1348   guchar r, g, b;
1349
1350   bptr = buf;
1351   bpl = image->bpl;
1352   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1353   for (y = 0; y < height; y++)
1354     {
1355       bp2 = bptr;
1356       obptr = obuf;
1357       if (((unsigned long)obuf | (unsigned long) bp2) & 3)
1358         {
1359           for (x = 0; x < width; x++)
1360             {
1361               r = *bp2++;
1362               g = *bp2++;
1363               b = *bp2++;
1364               ((guint16 *)obptr)[0] = ((r & 0xf8) << 8) |
1365                 ((g & 0xfc) << 3) |
1366                 (b >> 3);
1367               obptr += 2;
1368             }
1369         }
1370       else
1371         {
1372           for (x = 0; x < width - 3; x += 4)
1373             {
1374               guint32 r1b0g0r0;
1375               guint32 g2r2b1g1;
1376               guint32 b3g3r3b2;
1377
1378               r1b0g0r0 = ((guint32 *)bp2)[0];
1379               g2r2b1g1 = ((guint32 *)bp2)[1];
1380               b3g3r3b2 = ((guint32 *)bp2)[2];
1381               ((guint32 *)obptr)[0] =
1382                 ((r1b0g0r0 & 0xf8) << 8) |
1383                 ((r1b0g0r0 & 0xfc00) >> 5) |
1384                 ((r1b0g0r0 & 0xf80000) >> 19) |
1385                 (r1b0g0r0 & 0xf8000000) |
1386                 ((g2r2b1g1 & 0xfc) << 19) |
1387                 ((g2r2b1g1 & 0xf800) << 5);
1388               ((guint32 *)obptr)[1] =
1389                 ((g2r2b1g1 & 0xf80000) >> 8) |
1390                 ((g2r2b1g1 & 0xfc000000) >> 21) |
1391                 ((b3g3r3b2 & 0xf8) >> 3) |
1392                 ((b3g3r3b2 & 0xf800) << 16) |
1393                 ((b3g3r3b2 & 0xfc0000) << 3) |
1394                 ((b3g3r3b2 & 0xf8000000) >> 11);
1395               bp2 += 12;
1396               obptr += 8;
1397             }
1398           for (; x < width; x++)
1399             {
1400               r = *bp2++;
1401               g = *bp2++;
1402               b = *bp2++;
1403               ((guint16 *)obptr)[0] = ((r & 0xf8) << 8) |
1404                 ((g & 0xfc) << 3) |
1405                 (b >> 3);
1406               obptr += 2;
1407             }
1408         }
1409       bptr += rowstride;
1410       obuf += bpl;
1411     }
1412 }
1413 #else
1414 /* Render a 24-bit RGB image in buf into the GdkImage, without dithering.
1415    This assumes native byte ordering - what should really be done is to
1416    check whether the image byte_order is consistent with the _ENDIAN
1417    config flag, and if not, use a different function.
1418
1419    This routine is faster than the one included with Gtk 1.0 for a number
1420    of reasons:
1421
1422    1. Shifting instead of lookup tables (less memory traffic).
1423
1424    2. Much less register pressure, especially because shifts are
1425    in the code.
1426
1427    3. A memcpy is avoided (i.e. the transfer function).
1428
1429    4. On big-endian architectures, byte swapping is avoided.
1430
1431    That said, it wouldn't be hard to make it even faster - just make an
1432    inner loop that reads 3 words (i.e. 4 24-bit pixels), does a lot of
1433    shifting and masking, then writes 2 words.
1434 */
1435 static void
1436 gdk_rgb_convert_565 (GdkRgbInfo *image_info, GdkImage *image,
1437                      gint x0, gint y0, gint width, gint height,
1438                      guchar *buf, int rowstride,
1439                      gint x_align, gint y_align, GdkRgbCmap *cmap)
1440 {
1441   int x, y;
1442   guchar *obuf;
1443   gint bpl;
1444   guchar *bptr, *bp2;
1445   guchar r, g, b;
1446
1447   bptr = buf;
1448   bpl = image->bpl;
1449   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1450   for (y = 0; y < height; y++)
1451     {
1452       bp2 = bptr;
1453       for (x = 0; x < width; x++)
1454         {
1455           r = *bp2++;
1456           g = *bp2++;
1457           b = *bp2++;
1458           ((unsigned short *)obuf)[x] = ((r & 0xf8) << 8) |
1459             ((g & 0xfc) << 3) |
1460             (b >> 3);
1461         }
1462       bptr += rowstride;
1463       obuf += bpl;
1464     }
1465 }
1466 #endif
1467
1468 #ifdef HAIRY_CONVERT_565
1469 static void
1470 gdk_rgb_convert_565_gray (GdkRgbInfo *image_info, GdkImage *image,
1471                           gint x0, gint y0, gint width, gint height,
1472                           guchar *buf, int rowstride,
1473                           gint x_align, gint y_align, GdkRgbCmap *cmap)
1474 {
1475   int x, y;
1476   guchar *obuf, *obptr;
1477   gint bpl;
1478   guchar *bptr, *bp2;
1479   guchar g;
1480
1481   bptr = buf;
1482   bpl = image->bpl;
1483   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1484   for (y = 0; y < height; y++)
1485     {
1486       bp2 = bptr;
1487       obptr = obuf;
1488       if (((unsigned long)obuf | (unsigned long) bp2) & 3)
1489         {
1490           for (x = 0; x < width; x++)
1491             {
1492               g = *bp2++;
1493               ((guint16 *)obptr)[0] = ((g & 0xf8) << 8) |
1494                 ((g & 0xfc) << 3) |
1495                 (g >> 3);
1496               obptr += 2;
1497             }
1498         }
1499       else
1500         {
1501           for (x = 0; x < width - 3; x += 4)
1502             {
1503               guint32 g3g2g1g0;
1504
1505               g3g2g1g0 = ((guint32 *)bp2)[0];
1506               ((guint32 *)obptr)[0] =
1507                 ((g3g2g1g0 & 0xf8) << 8) |
1508                 ((g3g2g1g0 & 0xfc) << 3) |
1509                 ((g3g2g1g0 & 0xf8) >> 3) |
1510                 (g3g2g1g0 & 0xf800) << 16 |
1511                 ((g3g2g1g0 & 0xfc00) << 11) |
1512                 ((g3g2g1g0 & 0xf800) << 5);
1513               ((guint32 *)obptr)[1] =
1514                 ((g3g2g1g0 & 0xf80000) >> 8) |
1515                 ((g3g2g1g0 & 0xfc0000) >> 13) |
1516                 ((g3g2g1g0 & 0xf80000) >> 19) |
1517                 (g3g2g1g0 & 0xf8000000) |
1518                 ((g3g2g1g0 & 0xfc000000) >> 5) |
1519                 ((g3g2g1g0 & 0xf8000000) >> 11);
1520               bp2 += 4;
1521               obptr += 8;
1522             }
1523           for (; x < width; x++)
1524             {
1525               g = *bp2++;
1526               ((guint16 *)obptr)[0] = ((g & 0xf8) << 8) |
1527                 ((g & 0xfc) << 3) |
1528                 (g >> 3);
1529               obptr += 2;
1530             }
1531         }
1532       bptr += rowstride;
1533       obuf += bpl;
1534     }
1535 }
1536 #else
1537 static void
1538 gdk_rgb_convert_565_gray (GdkRgbInfo *image_info, GdkImage *image,
1539                           gint x0, gint y0, gint width, gint height,
1540                           guchar *buf, int rowstride,
1541                           gint x_align, gint y_align, GdkRgbCmap *cmap)
1542 {
1543   int x, y;
1544   guchar *obuf;
1545   gint bpl;
1546   guchar *bptr, *bp2;
1547   guchar g;
1548
1549   bptr = buf;
1550   bpl = image->bpl;
1551   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1552   for (y = 0; y < height; y++)
1553     {
1554       bp2 = bptr;
1555       for (x = 0; x < width; x++)
1556         {
1557           g = *bp2++;
1558           ((guint16 *)obuf)[x] = ((g & 0xf8) << 8) |
1559             ((g & 0xfc) << 3) |
1560             (g >> 3);
1561         }
1562       bptr += rowstride;
1563       obuf += bpl;
1564     }
1565 }
1566 #endif
1567
1568 static void
1569 gdk_rgb_convert_565_br (GdkRgbInfo *image_info, GdkImage *image,
1570                         gint x0, gint y0, gint width, gint height,
1571                         guchar *buf, int rowstride,
1572                         gint x_align, gint y_align, GdkRgbCmap *cmap)
1573 {
1574   int x, y;
1575   guchar *obuf;
1576   gint bpl;
1577   guchar *bptr, *bp2;
1578   guchar r, g, b;
1579
1580   bptr = buf;
1581   bpl = image->bpl;
1582   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1583   for (y = 0; y < height; y++)
1584     {
1585       bp2 = bptr;
1586       for (x = 0; x < width; x++)
1587         {
1588           r = *bp2++;
1589           g = *bp2++;
1590           b = *bp2++;
1591           /* final word is:
1592              g4 g3 g2 b7 b6 b5 b4 b3  r7 r6 r5 r4 r3 g7 g6 g5
1593            */
1594           ((unsigned short *)obuf)[x] = (r & 0xf8) |
1595             ((g & 0xe0) >> 5) |
1596             ((g & 0x1c) << 11) |
1597             ((b & 0xf8) << 5);
1598         }
1599       bptr += rowstride;
1600       obuf += bpl;
1601     }
1602 }
1603
1604 /* Thanks to Ray Lehtiniemi for a patch that resulted in a ~25% speedup
1605    in this mode. */
1606 #ifdef HAIRY_CONVERT_565
1607 static void
1608 gdk_rgb_convert_565_d (GdkRgbInfo *image_info, GdkImage *image,
1609                      gint x0, gint y0, gint width, gint height,
1610                      guchar *buf, int rowstride,
1611                      gint x_align, gint y_align, GdkRgbCmap *cmap)
1612 {
1613   /* Now this is what I'd call some highly tuned code! */
1614   int x, y;
1615   guchar *obuf, *obptr;
1616   gint bpl;
1617   guchar *bptr, *bp2;
1618
1619   width += x_align;
1620   height += y_align;
1621   
1622   bptr = buf;
1623   bpl = image->bpl;
1624   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1625   for (y = y_align; y < height; y++)
1626     {
1627       guint32 *dmp = DM_565 + ((y & (DM_HEIGHT - 1)) << DM_WIDTH_SHIFT);
1628       bp2 = bptr;
1629       obptr = obuf;
1630       if (((unsigned long)obuf | (unsigned long) bp2) & 3)
1631         {
1632           for (x = x_align; x < width; x++)
1633             {
1634               gint32 rgb = *bp2++ << 20;
1635               rgb += *bp2++ << 10;
1636               rgb += *bp2++;
1637               rgb += dmp[x & (DM_WIDTH - 1)];
1638               rgb += 0x10040100
1639                 - ((rgb & 0x1e0001e0) >> 5)
1640                 - ((rgb & 0x00070000) >> 6);
1641
1642               ((unsigned short *)obptr)[0] =
1643                 ((rgb & 0x0f800000) >> 12) |
1644                 ((rgb & 0x0003f000) >> 7) |
1645                 ((rgb & 0x000000f8) >> 3);
1646               obptr += 2;
1647             }
1648         }
1649       else
1650         {
1651           for (x = x_align; x < width - 3; x += 4)
1652             {
1653               guint32 r1b0g0r0;
1654               guint32 g2r2b1g1;
1655               guint32 b3g3r3b2;
1656               guint32 rgb02, rgb13;
1657
1658               r1b0g0r0 = ((guint32 *)bp2)[0];
1659               g2r2b1g1 = ((guint32 *)bp2)[1];
1660               b3g3r3b2 = ((guint32 *)bp2)[2];
1661               rgb02 =
1662                 ((r1b0g0r0 & 0xff) << 20) +
1663                 ((r1b0g0r0 & 0xff00) << 2) +
1664                 ((r1b0g0r0 & 0xff0000) >> 16) +
1665                 dmp[x & (DM_WIDTH - 1)];
1666               rgb02 += 0x10040100
1667                 - ((rgb02 & 0x1e0001e0) >> 5)
1668                 - ((rgb02 & 0x00070000) >> 6);
1669               rgb13 =
1670                 ((r1b0g0r0 & 0xff000000) >> 4) +
1671                 ((g2r2b1g1 & 0xff) << 10) +
1672                 ((g2r2b1g1 & 0xff00) >> 8) +
1673                 dmp[(x + 1) & (DM_WIDTH - 1)];
1674               rgb13 += 0x10040100
1675                 - ((rgb13 & 0x1e0001e0) >> 5)
1676                 - ((rgb13 & 0x00070000) >> 6);
1677               ((guint32 *)obptr)[0] =
1678                 ((rgb02 & 0x0f800000) >> 12) |
1679                 ((rgb02 & 0x0003f000) >> 7) |
1680                 ((rgb02 & 0x000000f8) >> 3) |
1681                 ((rgb13 & 0x0f800000) << 4) |
1682                 ((rgb13 & 0x0003f000) << 9) |
1683                 ((rgb13 & 0x000000f8) << 13);
1684               rgb02 =
1685                 ((g2r2b1g1 & 0xff0000) << 4) +
1686                 ((g2r2b1g1 & 0xff000000) >> 14) +
1687                 (b3g3r3b2 & 0xff) +
1688                 dmp[(x + 2) & (DM_WIDTH - 1)];
1689               rgb02 += 0x10040100
1690                 - ((rgb02 & 0x1e0001e0) >> 5)
1691                 - ((rgb02 & 0x00070000) >> 6);
1692               rgb13 =
1693                 ((b3g3r3b2 & 0xff00) << 12) +
1694                 ((b3g3r3b2 & 0xff0000) >> 6) +
1695                 ((b3g3r3b2 & 0xff000000) >> 24) +
1696                 dmp[(x + 3) & (DM_WIDTH - 1)];
1697               rgb13 += 0x10040100
1698                 - ((rgb13 & 0x1e0001e0) >> 5)
1699                 - ((rgb13 & 0x00070000) >> 6);
1700               ((guint32 *)obptr)[1] =
1701                 ((rgb02 & 0x0f800000) >> 12) |
1702                 ((rgb02 & 0x0003f000) >> 7) |
1703                 ((rgb02 & 0x000000f8) >> 3) |
1704                 ((rgb13 & 0x0f800000) << 4) |
1705                 ((rgb13 & 0x0003f000) << 9) |
1706                 ((rgb13 & 0x000000f8) << 13);
1707               bp2 += 12;
1708               obptr += 8;
1709             }
1710           for (; x < width; x++)
1711             {
1712               gint32 rgb = *bp2++ << 20;
1713               rgb += *bp2++ << 10;
1714               rgb += *bp2++;
1715               rgb += dmp[x & (DM_WIDTH - 1)];
1716               rgb += 0x10040100
1717                 - ((rgb & 0x1e0001e0) >> 5)
1718                 - ((rgb & 0x00070000) >> 6);
1719
1720               ((unsigned short *)obptr)[0] =
1721                 ((rgb & 0x0f800000) >> 12) |
1722                 ((rgb & 0x0003f000) >> 7) |
1723                 ((rgb & 0x000000f8) >> 3);
1724               obptr += 2;
1725             }
1726         }
1727       bptr += rowstride;
1728       obuf += bpl;
1729     }
1730 }
1731 #else
1732 static void
1733 gdk_rgb_convert_565_d (GdkRgbInfo *image_info, GdkImage *image,
1734                        gint x0, gint y0, gint width, gint height,
1735                        guchar *buf, int rowstride,
1736                        gint x_align, gint y_align, GdkRgbCmap *cmap)
1737 {
1738   int x, y;
1739   guchar *obuf;
1740   gint bpl;
1741   guchar *bptr;
1742
1743   width += x_align;
1744   height += y_align;
1745   
1746   bptr = buf;
1747   bpl = image->bpl;
1748   obuf = ((guchar *)image->mem) + y0 * bpl + (x0 - x_align) * 2;
1749
1750   for (y = y_align; y < height; y++)
1751     {
1752       guint32 *dmp = DM_565 + ((y & (DM_HEIGHT - 1)) << DM_WIDTH_SHIFT);
1753       guchar *bp2 = bptr;
1754
1755       for (x = x_align; x < width; x++)
1756         {
1757           gint32 rgb = *bp2++ << 20;
1758           rgb += *bp2++ << 10;
1759           rgb += *bp2++;
1760           rgb += dmp[x & (DM_WIDTH - 1)];
1761           rgb += 0x10040100
1762             - ((rgb & 0x1e0001e0) >> 5)
1763             - ((rgb & 0x00070000) >> 6);
1764
1765           ((unsigned short *)obuf)[x] =
1766             ((rgb & 0x0f800000) >> 12) |
1767             ((rgb & 0x0003f000) >> 7) |
1768             ((rgb & 0x000000f8) >> 3);
1769         }
1770
1771       bptr += rowstride;
1772       obuf += bpl;
1773     }
1774 }
1775 #endif
1776
1777 static void
1778 gdk_rgb_convert_555 (GdkRgbInfo *image_info, GdkImage *image,
1779                      gint x0, gint y0, gint width, gint height,
1780                      guchar *buf, int rowstride,
1781                      gint x_align, gint y_align, GdkRgbCmap *cmap)
1782 {
1783   int x, y;
1784   guchar *obuf;
1785   gint bpl;
1786   guchar *bptr, *bp2;
1787   guchar r, g, b;
1788
1789   bptr = buf;
1790   bpl = image->bpl;
1791   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1792   for (y = 0; y < height; y++)
1793     {
1794       bp2 = bptr;
1795       for (x = 0; x < width; x++)
1796         {
1797           r = *bp2++;
1798           g = *bp2++;
1799           b = *bp2++;
1800           ((unsigned short *)obuf)[x] = ((r & 0xf8) << 7) |
1801             ((g & 0xf8) << 2) |
1802             (b >> 3);
1803         }
1804       bptr += rowstride;
1805       obuf += bpl;
1806     }
1807 }
1808
1809 static void
1810 gdk_rgb_convert_555_br (GdkRgbInfo *image_info, GdkImage *image,
1811                         gint x0, gint y0, gint width, gint height,
1812                         guchar *buf, int rowstride,
1813                         gint x_align, gint y_align, GdkRgbCmap *cmap)
1814 {
1815   int x, y;
1816   guchar *obuf;
1817   gint bpl;
1818   guchar *bptr, *bp2;
1819   guchar r, g, b;
1820
1821   bptr = buf;
1822   bpl = image->bpl;
1823   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1824   for (y = 0; y < height; y++)
1825     {
1826       bp2 = bptr;
1827       for (x = 0; x < width; x++)
1828         {
1829           r = *bp2++;
1830           g = *bp2++;
1831           b = *bp2++;
1832           /* final word is:
1833              g5 g4 g3 b7 b6 b5 b4 b3  0 r7 r6 r5 r4 r3 g7 g6
1834            */
1835           ((unsigned short *)obuf)[x] = ((r & 0xf8) >> 1) |
1836             ((g & 0xc0) >> 6) |
1837             ((g & 0x38) << 10) |
1838             ((b & 0xf8) << 5);
1839         }
1840       bptr += rowstride;
1841       obuf += bpl;
1842     }
1843 }
1844
1845 static void
1846 gdk_rgb_convert_888_msb (GdkRgbInfo *image_info, GdkImage *image,
1847                          gint x0, gint y0, gint width, gint height,
1848                          guchar *buf, int rowstride,
1849                          gint x_align, gint y_align, GdkRgbCmap *cmap)
1850 {
1851   int y;
1852   guchar *obuf;
1853   gint bpl;
1854   guchar *bptr;
1855
1856   bptr = buf;
1857   bpl = image->bpl;
1858   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 3;
1859   for (y = 0; y < height; y++)
1860     {
1861       memcpy (obuf, bptr, width + width + width);
1862       bptr += rowstride;
1863       obuf += bpl;
1864     }
1865 }
1866
1867 /* todo: optimize this */
1868 #if G_BYTE_ORDER == G_LITTLE_ENDIAN
1869 #define HAIRY_CONVERT_888
1870 #endif
1871
1872 #ifdef HAIRY_CONVERT_888
1873 static void
1874 gdk_rgb_convert_888_lsb (GdkRgbInfo *image_info, GdkImage *image,
1875                          gint x0, gint y0, gint width, gint height,
1876                          guchar *buf, int rowstride,
1877                          gint x_align, gint y_align, GdkRgbCmap *cmap)
1878 {
1879   int x, y;
1880   guchar *obuf, *obptr;
1881   gint bpl;
1882   guchar *bptr, *bp2;
1883   int r, g, b;
1884
1885   bptr = buf;
1886   bpl = image->bpl;
1887   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 3;
1888   for (y = 0; y < height; y++)
1889     {
1890       bp2 = bptr;
1891       obptr = obuf;
1892       if (((unsigned long)obuf | (unsigned long) bp2) & 3)
1893         {
1894           for (x = 0; x < width; x++)
1895             {
1896               r = bp2[0];
1897               g = bp2[1];
1898               b = bp2[2];
1899               *obptr++ = b;
1900               *obptr++ = g;
1901               *obptr++ = r;
1902               bp2 += 3;
1903             }
1904         }
1905       else
1906         {
1907           for (x = 0; x < width - 3; x += 4)
1908             {
1909               guint32 r1b0g0r0;
1910               guint32 g2r2b1g1;
1911               guint32 b3g3r3b2;
1912
1913               r1b0g0r0 = ((guint32 *)bp2)[0];
1914               g2r2b1g1 = ((guint32 *)bp2)[1];
1915               b3g3r3b2 = ((guint32 *)bp2)[2];
1916               ((guint32 *)obptr)[0] =
1917                 (r1b0g0r0 & 0xff00) |
1918                 ((r1b0g0r0 & 0xff0000) >> 16) |
1919                 (((g2r2b1g1 & 0xff00) | (r1b0g0r0 & 0xff)) << 16);
1920               ((guint32 *)obptr)[1] =
1921                 (g2r2b1g1 & 0xff0000ff) |
1922                 ((r1b0g0r0 & 0xff000000) >> 16) |
1923                 ((b3g3r3b2 & 0xff) << 16);
1924               ((guint32 *)obptr)[2] =
1925                 (((g2r2b1g1 & 0xff0000) | (b3g3r3b2 & 0xff000000)) >> 16) |
1926                 ((b3g3r3b2 & 0xff00) << 16) |
1927                 ((b3g3r3b2 & 0xff0000));
1928               bp2 += 12;
1929               obptr += 12;
1930             }
1931           for (; x < width; x++)
1932             {
1933               r = bp2[0];
1934               g = bp2[1];
1935               b = bp2[2];
1936               *obptr++ = b;
1937               *obptr++ = g;
1938               *obptr++ = r;
1939               bp2 += 3;
1940             }
1941         }
1942       bptr += rowstride;
1943       obuf += bpl;
1944     }
1945 }
1946 #else
1947 static void
1948 gdk_rgb_convert_888_lsb (GdkRgbInfo *image_info, GdkImage *image,
1949                          gint x0, gint y0, gint width, gint height,
1950                          guchar *buf, int rowstride,
1951                          gint x_align, gint y_align, GdkRgbCmap *cmap)
1952 {
1953   int x, y;
1954   guchar *obuf;
1955   gint bpl;
1956   guchar *bptr, *bp2;
1957   int r, g, b;
1958
1959   bptr = buf;
1960   bpl = image->bpl;
1961   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 3;
1962   for (y = 0; y < height; y++)
1963     {
1964       bp2 = bptr;
1965       for (x = 0; x < width; x++)
1966         {
1967           r = bp2[0];
1968           g = bp2[1];
1969           b = bp2[2];
1970           obuf[x * 3] = b;
1971           obuf[x * 3 + 1] = g;
1972           obuf[x * 3 + 2] = r;
1973           bp2 += 3;
1974         }
1975       bptr += rowstride;
1976       obuf += bpl;
1977     }
1978 }
1979 #endif
1980
1981 /* convert 24-bit packed to 32-bit unpacked */
1982 /* todo: optimize this */
1983 static void
1984 gdk_rgb_convert_0888 (GdkRgbInfo *image_info, GdkImage *image,
1985                       gint x0, gint y0, gint width, gint height,
1986                       guchar *buf, int rowstride,
1987                       gint x_align, gint y_align, GdkRgbCmap *cmap)
1988 {
1989   int x, y;
1990   guchar *obuf;
1991   gint bpl;
1992   guchar *bptr, *bp2;
1993   int r, g, b;
1994
1995   bptr = buf;
1996   bpl = image->bpl;
1997   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 4;
1998   for (y = 0; y < height; y++)
1999     {
2000       bp2 = bptr;
2001       for (x = 0; x < width; x++)
2002         {
2003           r = bp2[0];
2004           g = bp2[1];
2005           b = bp2[2];
2006           ((guint32 *)obuf)[x] = (r << 16) | (g << 8) | b;
2007           bp2 += 3;
2008         }
2009       bptr += rowstride;
2010       obuf += bpl;
2011     }
2012 }
2013
2014 static void
2015 gdk_rgb_convert_0888_br (GdkRgbInfo *image_info, GdkImage *image,
2016                          gint x0, gint y0, gint width, gint height,
2017                          guchar *buf, int rowstride,
2018                          gint x_align, gint y_align, GdkRgbCmap *cmap)
2019 {
2020   int x, y;
2021   guchar *obuf;
2022   gint bpl;
2023   guchar *bptr, *bp2;
2024   int r, g, b;
2025
2026   bptr = buf;
2027   bpl = image->bpl;
2028   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 4;
2029   for (y = 0; y < height; y++)
2030     {
2031       bp2 = bptr;
2032       for (x = 0; x < width; x++)
2033         {
2034           r = bp2[0];
2035           g = bp2[1];
2036           b = bp2[2];
2037           ((guint32 *)obuf)[x] = (b << 24) | (g << 16) | (r << 8);
2038           bp2 += 3;
2039         }
2040       bptr += rowstride;
2041       obuf += bpl;
2042     }
2043 }
2044
2045 static void
2046 gdk_rgb_convert_8880_br (GdkRgbInfo *image_info, GdkImage *image,
2047                          gint x0, gint y0, gint width, gint height,
2048                          guchar *buf, int rowstride,
2049                          gint x_align, gint y_align, GdkRgbCmap *cmap)
2050 {
2051   int x, y;
2052   guchar *obuf;
2053   gint bpl;
2054   guchar *bptr, *bp2;
2055   int r, g, b;
2056
2057   bptr = buf;
2058   bpl = image->bpl;
2059   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 4;
2060   for (y = 0; y < height; y++)
2061     {
2062       bp2 = bptr;
2063       for (x = 0; x < width; x++)
2064         {
2065           r = bp2[0];
2066           g = bp2[1];
2067           b = bp2[2];
2068           ((guint32 *)obuf)[x] = (b << 16) | (g << 8) | r;
2069           bp2 += 3;
2070         }
2071       bptr += rowstride;
2072       obuf += bpl;
2073     }
2074 }
2075
2076 /* Generic truecolor/directcolor conversion function. Slow, but these
2077    are oddball modes. */
2078 static void
2079 gdk_rgb_convert_truecolor_lsb (GdkRgbInfo *image_info, GdkImage *image,
2080                                gint x0, gint y0, gint width, gint height,
2081                                guchar *buf, int rowstride,
2082                                gint x_align, gint y_align,
2083                                GdkRgbCmap *cmap)
2084 {
2085   int x, y;
2086   guchar *obuf, *obptr;
2087   gint bpl;
2088   guchar *bptr, *bp2;
2089   gint r, g, b;
2090   gint r_right, r_left;
2091   gint g_right, g_left;
2092   gint b_right, b_left;
2093   gint bpp;
2094   guint32 pixel;
2095   gint i;
2096
2097   r_right = 8 - image_info->visual->red_prec;
2098   r_left = image_info->visual->red_shift;
2099   g_right = 8 - image_info->visual->green_prec;
2100   g_left = image_info->visual->green_shift;
2101   b_right = 8 - image_info->visual->blue_prec;
2102   b_left = image_info->visual->blue_shift;
2103   bpp = image_info->bpp;
2104   bptr = buf;
2105   bpl = image->bpl;
2106   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * bpp;
2107   for (y = 0; y < height; y++)
2108     {
2109       obptr = obuf;
2110       bp2 = bptr;
2111       for (x = 0; x < width; x++)
2112         {
2113           r = bp2[0];
2114           g = bp2[1];
2115           b = bp2[2];
2116           pixel = ((r >> r_right) << r_left) |
2117             ((g >> g_right) << g_left) |
2118             ((b >> b_right) << b_left);
2119           for (i = 0; i < bpp; i++)
2120             {
2121               *obptr++ = pixel & 0xff;
2122               pixel >>= 8;
2123             }
2124           bp2 += 3;
2125         }
2126       bptr += rowstride;
2127       obuf += bpl;
2128     }
2129 }
2130
2131 static void
2132 gdk_rgb_convert_truecolor_lsb_d (GdkRgbInfo *image_info, GdkImage *image,
2133                                  gint x0, gint y0, gint width, gint height,
2134                                  guchar *buf, int rowstride,
2135                                  gint x_align, gint y_align,
2136                                  GdkRgbCmap *cmap)
2137 {
2138   int x, y;
2139   guchar *obuf, *obptr;
2140   gint bpl;
2141   guchar *bptr, *bp2;
2142   gint r, g, b;
2143   gint r_right, r_left, r_prec;
2144   gint g_right, g_left, g_prec;
2145   gint b_right, b_left, b_prec;
2146   gint bpp;
2147   guint32 pixel;
2148   gint i;
2149   gint dith;
2150   gint r1, g1, b1;
2151   const guchar *dmp;
2152
2153   r_right = 8 - image_info->visual->red_prec;
2154   r_left = image_info->visual->red_shift;
2155   r_prec = image_info->visual->red_prec;
2156   g_right = 8 - image_info->visual->green_prec;
2157   g_left = image_info->visual->green_shift;
2158   g_prec = image_info->visual->green_prec;
2159   b_right = 8 - image_info->visual->blue_prec;
2160   b_left = image_info->visual->blue_shift;
2161   b_prec = image_info->visual->blue_prec;
2162   bpp = image_info->bpp;
2163   bptr = buf;
2164   bpl = image->bpl;
2165   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * bpp;
2166   for (y = 0; y < height; y++)
2167     {
2168       dmp = DM[(y_align + y) & (DM_HEIGHT - 1)];
2169       obptr = obuf;
2170       bp2 = bptr;
2171       for (x = 0; x < width; x++)
2172         {
2173           r = bp2[0];
2174           g = bp2[1];
2175           b = bp2[2];
2176           dith = dmp[(x_align + x) & (DM_WIDTH - 1)] << 2;
2177           r1 = r + (dith >> r_prec);
2178           g1 = g + ((252 - dith) >> g_prec);
2179           b1 = b + (dith >> b_prec);
2180           pixel = (((r1 - (r1 >> r_prec)) >> r_right) << r_left) |
2181             (((g1 - (g1 >> g_prec)) >> g_right) << g_left) |
2182             (((b1 - (b1 >> b_prec)) >> b_right) << b_left);
2183           for (i = 0; i < bpp; i++)
2184             {
2185               *obptr++ = pixel & 0xff;
2186               pixel >>= 8;
2187             }
2188           bp2 += 3;
2189         }
2190       bptr += rowstride;
2191       obuf += bpl;
2192     }
2193 }
2194
2195 static void
2196 gdk_rgb_convert_truecolor_msb (GdkRgbInfo *image_info, GdkImage *image,
2197                                gint x0, gint y0, gint width, gint height,
2198                                guchar *buf, int rowstride,
2199                                gint x_align, gint y_align,
2200                                GdkRgbCmap *cmap)
2201 {
2202   int x, y;
2203   guchar *obuf, *obptr;
2204   gint bpl;
2205   guchar *bptr, *bp2;
2206   gint r, g, b;
2207   gint r_right, r_left;
2208   gint g_right, g_left;
2209   gint b_right, b_left;
2210   gint bpp;
2211   guint32 pixel;
2212   gint shift, shift_init;
2213
2214   r_right = 8 - image_info->visual->red_prec;
2215   r_left = image_info->visual->red_shift;
2216   g_right = 8 - image_info->visual->green_prec;
2217   g_left = image_info->visual->green_shift;
2218   b_right = 8 - image_info->visual->blue_prec;
2219   b_left = image_info->visual->blue_shift;
2220   bpp = image_info->bpp;
2221   bptr = buf;
2222   bpl = image->bpl;
2223   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * bpp;
2224   shift_init = (bpp - 1) << 3;
2225   for (y = 0; y < height; y++)
2226     {
2227       obptr = obuf;
2228       bp2 = bptr;
2229       for (x = 0; x < width; x++)
2230         {
2231           r = bp2[0];
2232           g = bp2[1];
2233           b = bp2[2];
2234           pixel = ((r >> r_right) << r_left) |
2235             ((g >> g_right) << g_left) |
2236             ((b >> b_right) << b_left);
2237           for (shift = shift_init; shift >= 0; shift -= 8)
2238             {
2239               *obptr++ = (pixel >> shift) & 0xff;
2240             }
2241           bp2 += 3;
2242         }
2243       bptr += rowstride;
2244       obuf += bpl;
2245     }
2246 }
2247
2248 static void
2249 gdk_rgb_convert_truecolor_msb_d (GdkRgbInfo *image_info, GdkImage *image,
2250                                  gint x0, gint y0, gint width, gint height,
2251                                  guchar *buf, int rowstride,
2252                                  gint x_align, gint y_align,
2253                                  GdkRgbCmap *cmap)
2254 {
2255   int x, y;
2256   guchar *obuf, *obptr;
2257   gint bpl;
2258   guchar *bptr, *bp2;
2259   gint r, g, b;
2260   gint r_right, r_left, r_prec;
2261   gint g_right, g_left, g_prec;
2262   gint b_right, b_left, b_prec;
2263   gint bpp;
2264   guint32 pixel;
2265   gint shift, shift_init;
2266   gint dith;
2267   gint r1, g1, b1;
2268   const guchar *dmp;
2269
2270   r_right = 8 - image_info->visual->red_prec;
2271   r_left = image_info->visual->red_shift;
2272   r_prec = image_info->visual->red_prec;
2273   g_right = 8 - image_info->visual->green_prec;
2274   g_left = image_info->visual->green_shift;
2275   g_prec = image_info->visual->green_prec;
2276   b_right = 8 - image_info->visual->blue_prec;
2277   b_left = image_info->visual->blue_shift;
2278   b_prec = image_info->visual->blue_prec;
2279   bpp = image_info->bpp;
2280   bptr = buf;
2281   bpl = image->bpl;
2282   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * bpp;
2283   shift_init = (bpp - 1) << 3;
2284   for (y = 0; y < height; y++)
2285     {
2286       dmp = DM[(y_align + y) & (DM_HEIGHT - 1)];
2287       obptr = obuf;
2288       bp2 = bptr;
2289       for (x = 0; x < width; x++)
2290         {
2291           r = bp2[0];
2292           g = bp2[1];
2293           b = bp2[2];
2294           dith = dmp[(x_align + x) & (DM_WIDTH - 1)] << 2;
2295           r1 = r + (dith >> r_prec);
2296           g1 = g + ((252 - dith) >> g_prec);
2297           b1 = b + (dith >> b_prec);
2298           pixel = (((r1 - (r1 >> r_prec)) >> r_right) << r_left) |
2299             (((g1 - (g1 >> g_prec)) >> g_right) << g_left) |
2300             (((b1 - (b1 >> b_prec)) >> b_right) << b_left);
2301           for (shift = shift_init; shift >= 0; shift -= 8)
2302             {
2303               *obptr++ = (pixel >> shift) & 0xff;
2304             }
2305           bp2 += 3;
2306         }
2307       bptr += rowstride;
2308       obuf += bpl;
2309     }
2310 }
2311
2312 /* This actually works for depths from 3 to 7 */
2313 static void
2314 gdk_rgb_convert_4 (GdkRgbInfo *image_info, GdkImage *image,
2315                    gint x0, gint y0, gint width, gint height,
2316                    guchar *buf, int rowstride,
2317                    gint x_align, gint y_align,
2318                    GdkRgbCmap *cmap)
2319 {
2320   int x, y;
2321   gint bpl;
2322   guchar *obuf, *obptr;
2323   guchar *bptr, *bp2;
2324   gint r, g, b;
2325   const guchar *dmp;
2326   gint dith;
2327   guchar *colorcube_d = image_info->colorcube_d;
2328   
2329   bptr = buf;
2330   bpl = image->bpl;
2331   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
2332   for (y = 0; y < height; y++)
2333     {
2334       dmp = DM[(y_align + y) & (DM_HEIGHT - 1)];
2335       bp2 = bptr;
2336       obptr = obuf;
2337       for (x = 0; x < width; x += 1)
2338         {
2339           r = *bp2++;
2340           g = *bp2++;
2341           b = *bp2++;
2342           dith = (dmp[(x_align + x) & (DM_WIDTH - 1)] << 2) | 3;
2343           obptr[0] = colorcube_d[(((r + dith) & 0x100) >> 2) |
2344                                 (((g + 258 - dith) & 0x100) >> 5) |
2345                                 (((b + dith) & 0x100) >> 8)];
2346           obptr++;
2347         }
2348       bptr += rowstride;
2349       obuf += bpl;
2350     }
2351 }
2352
2353 static void
2354 gdk_rgb_convert_4_pack (GdkRgbInfo *image_info, GdkImage *image,
2355                         gint x0, gint y0, gint width, gint height,
2356                         guchar *buf, int rowstride,
2357                         gint x_align, gint y_align,
2358                         GdkRgbCmap *cmap)
2359 {
2360   int x, y, ix;
2361   gint bpl;
2362   guchar *obuf, *obptr;
2363   guchar *bptr, *bp2;
2364   gint r, g, b;
2365   const guchar *dmp;
2366   gint dith;
2367   guchar *colorcube_d = image_info->colorcube_d;
2368   guchar pix0, pix1;
2369
2370   bptr = buf;
2371   bpl = image->bpl;
2372   obuf = ((guchar *)image->mem) + y0 * bpl + (x0 >> 1);
2373   for (y = 0; y < height; y++)
2374     {
2375       dmp = DM[(y_align + y) & (DM_HEIGHT - 1)];
2376       bp2 = bptr;
2377       obptr = obuf;
2378
2379       x = 0;
2380       if (x0 & 1)
2381         {
2382           r = *bp2++;
2383           g = *bp2++;
2384           b = *bp2++;
2385           dith = (dmp[(x_align + x + 1) & (DM_WIDTH - 1)] << 2) | 3;
2386           ix = (((r + dith) & 0x100) >> 2) |
2387             (((g + 258 - dith) & 0x100) >> 5) |
2388             (((b + dith) & 0x100) >> 8);
2389           pix1 = (colorcube_d[ix]);
2390           *obptr = (*obptr & 0xF0) | pix1;
2391           obptr++;
2392           x++;
2393         }
2394       while (x < width)
2395        {
2396          r = *bp2++;
2397          g = *bp2++;
2398          b = *bp2++;
2399          dith = (dmp[(x_align + x) & (DM_WIDTH - 1)] << 2) | 3;
2400          ix = (((r + dith) & 0x100) >> 2) |
2401            (((g + 258 - dith) & 0x100) >> 5) |
2402            (((b + dith) & 0x100) >> 8);
2403          pix0 = (colorcube_d[ix]);
2404          x++;
2405          if (x == width)
2406            pix1 = (*obptr & 0x0F);
2407          else
2408            {
2409              r = *bp2++;
2410              g = *bp2++;
2411              b = *bp2++;
2412              dith = (dmp[(x_align + x + 1) & (DM_WIDTH - 1)] << 2) | 3;
2413              ix = (((r + dith) & 0x100) >> 2) |
2414                (((g + 258 - dith) & 0x100) >> 5) |
2415                (((b + dith) & 0x100) >> 8);
2416              pix1 = (colorcube_d[ix]);
2417              x++;
2418            }
2419          *obptr++ = (pix0 << 4) | pix1;
2420        }
2421       bptr += rowstride;
2422       obuf += bpl;
2423     }
2424 }
2425
2426 /* This actually works for depths from 3 to 7 */
2427 static void
2428 gdk_rgb_convert_gray4 (GdkRgbInfo *image_info, GdkImage *image,
2429                        gint x0, gint y0, gint width, gint height,
2430                        guchar *buf, int rowstride,
2431                        gint x_align, gint y_align, GdkRgbCmap *cmap)
2432 {
2433   int x, y;
2434   gint bpl;
2435   guchar *obuf, *obptr;
2436   guchar *bptr, *bp2;
2437   gint r, g, b;
2438   gint shift;
2439
2440   bptr = buf;
2441   bpl = image->bpl;
2442   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
2443   shift = 9 - image_info->visual->depth;
2444   for (y = 0; y < height; y++)
2445     {
2446       bp2 = bptr;
2447       obptr = obuf;
2448       for (x = 0; x < width; x++)
2449         {
2450           r = *bp2++;
2451           g = *bp2++;
2452           b = *bp2++;
2453           obptr[0] = (g + ((b + r) >> 1)) >> shift;
2454           obptr++;
2455         }
2456       bptr += rowstride;
2457       obuf += bpl;
2458     }
2459 }
2460
2461 static void
2462 gdk_rgb_convert_gray4_pack (GdkRgbInfo *image_info, GdkImage *image,
2463                             gint x0, gint y0, gint width, gint height,
2464                             guchar *buf, int rowstride,
2465                             gint x_align, gint y_align, GdkRgbCmap *cmap)
2466 {
2467   int x, y;
2468   gint bpl;
2469   guchar *obuf, *obptr;
2470   guchar *bptr, *bp2;
2471   gint r, g, b;
2472   gint shift;
2473   guchar pix0, pix1;
2474   /* todo: this is hardcoded to big-endian. Make endian-agile. */
2475
2476   bptr = buf;
2477   bpl = image->bpl;
2478   obuf = ((guchar *)image->mem) + y0 * bpl + (x0 >> 1);
2479   shift = 9 - image_info->visual->depth;
2480   for (y = 0; y < height; y++)
2481     {
2482       bp2 = bptr;
2483       obptr = obuf;
2484
2485       x = 0;
2486       if (x0 & 1)
2487         {
2488           r = *bp2++;
2489           g = *bp2++;
2490           b = *bp2++;
2491           pix1 = (g + ((b + r) >> 1)) >> shift;
2492           *obptr = (*obptr & 0xF0) | pix1;
2493           obptr++;
2494           x++;
2495         }
2496       while (x < width)
2497         {
2498           r = *bp2++;
2499           g = *bp2++;
2500           b = *bp2++;
2501           pix0 = (g + ((b + r) >> 1)) >> shift;
2502           x++;
2503           if (x == width)
2504             pix1 = (*obptr & 0x0F);
2505           else
2506             {
2507               r = *bp2++;
2508               g = *bp2++;
2509               b = *bp2++;
2510               pix1 = (g + ((b + r) >> 1)) >> shift;
2511               x++;
2512             }
2513           *obptr++ = (pix0 << 4) | pix1;
2514         }
2515       bptr += rowstride;
2516       obuf += bpl;
2517     }
2518 }
2519
2520 /* This actually works for depths from 3 to 7 */
2521 static void
2522 gdk_rgb_convert_gray4_d (GdkRgbInfo *image_info, GdkImage *image,
2523                          gint x0, gint y0, gint width, gint height,
2524                          guchar *buf, int rowstride,
2525                          gint x_align, gint y_align, GdkRgbCmap *cmap)
2526 {
2527   int x, y;
2528   gint bpl;
2529   guchar *obuf, *obptr;
2530   guchar *bptr, *bp2;
2531   gint r, g, b;
2532   const guchar *dmp;
2533   gint prec, right;
2534   gint gray;
2535
2536   bptr = buf;
2537   bpl = image->bpl;
2538   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
2539   prec = image_info->visual->depth;
2540   right = 8 - prec;
2541   for (y = 0; y < height; y++)
2542     {
2543       bp2 = bptr;
2544       obptr = obuf;
2545       dmp = DM[(y_align + y) & (DM_HEIGHT - 1)];
2546       for (x = 0; x < width; x++)
2547         {
2548           r = *bp2++;
2549           g = *bp2++;
2550           b = *bp2++;
2551           gray = (g + ((b + r) >> 1)) >> 1;
2552           gray += (dmp[(x_align + x) & (DM_WIDTH - 1)] << 2) >> prec;
2553           obptr[0] = (gray - (gray >> prec)) >> right;
2554           obptr++;
2555         }
2556       bptr += rowstride;
2557       obuf += bpl;
2558     }
2559 }
2560
2561 static void
2562 gdk_rgb_convert_gray4_d_pack (GdkRgbInfo *image_info, GdkImage *image,
2563                               gint x0, gint y0, gint width, gint height,
2564                               guchar *buf, int rowstride,
2565                               gint x_align, gint y_align, GdkRgbCmap *cmap)
2566 {
2567   int x, y;
2568   gint bpl;
2569   guchar *obuf, *obptr;
2570   guchar *bptr, *bp2;
2571   gint r, g, b;
2572   const guchar *dmp;
2573   gint prec, right;
2574   gint gray;
2575   guchar pix0, pix1;
2576   /* todo: this is hardcoded to big-endian. Make endian-agile. */
2577
2578   bptr = buf;
2579   bpl = image->bpl;
2580   obuf = ((guchar *)image->mem) + y0 * bpl + (x0 >> 1);
2581   prec = image_info->visual->depth;
2582   right = 8 - prec;
2583   for (y = 0; y < height; y++)
2584     {
2585       bp2 = bptr;
2586       obptr = obuf;
2587       dmp = DM[(y_align + y) & (DM_HEIGHT - 1)];
2588
2589       x = 0;
2590       if (x0 & 1)
2591         {
2592           r = *bp2++;
2593           g = *bp2++;
2594           b = *bp2++;
2595           gray = (g + ((b + r) >> 1)) >> 1;
2596           gray += (dmp[(x_align + x + 1) & (DM_WIDTH - 1)] << 2) >> prec;
2597           pix1 = (gray - (gray >> prec)) >> right;
2598           *obptr = (*obptr & 0xF0) | pix1;
2599           obptr++;
2600           x++;
2601         }
2602       while (x < width)
2603         {
2604           r = *bp2++;
2605           g = *bp2++;
2606           b = *bp2++;
2607           gray = (g + ((b + r) >> 1)) >> 1;
2608           gray += (dmp[(x_align + x) & (DM_WIDTH - 1)] << 2) >> prec;
2609           pix0 = (gray - (gray >> prec)) >> right;
2610           x++;
2611           if (x == width)
2612             pix1 = (*obptr & 0x0F);
2613           else
2614             {
2615               r = *bp2++;
2616               g = *bp2++;
2617               b = *bp2++;
2618               gray = (g + ((b + r) >> 1)) >> 1;
2619               gray += (dmp[(x_align + x + 1) & (DM_WIDTH - 1)] << 2) >> prec;
2620               pix1 = (gray - (gray >> prec)) >> right;
2621               x++;
2622             }
2623           *obptr++ = (pix0 << 4) | pix1;
2624         }
2625       bptr += rowstride;
2626       obuf += bpl;
2627     }
2628 }
2629
2630 static void
2631 gdk_rgb_convert_1 (GdkRgbInfo *image_info, GdkImage *image,
2632                    gint x0, gint y0, gint width, gint height,
2633                    guchar *buf, int rowstride,
2634                    gint x_align, gint y_align,
2635                    GdkRgbCmap *cmap)
2636 {
2637   int x, y;
2638   gint bpl;
2639   guchar *obuf, *obptr;
2640   guchar *bptr, *bp2;
2641   gint r, g, b;
2642   const guchar *dmp;
2643   gint dith;
2644   guchar byte;
2645
2646   bptr = buf;
2647   bpl = image->bpl;
2648   obuf = ((guchar *)image->mem) + y0 * bpl + (x0 >> 3);
2649   byte = 0; /* unnecessary, but it keeps gcc from complaining */
2650   for (y = 0; y < height; y++)
2651     {
2652       dmp = DM[(y_align + y) & (DM_HEIGHT - 1)];
2653       bp2 = bptr;
2654       obptr = obuf;
2655       for (x = 0; x < width; x++)
2656         {
2657           r = *bp2++;
2658           g = *bp2++;
2659           b = *bp2++;
2660           dith = (dmp[(x_align + x) & (DM_WIDTH - 1)] << 4) | 4;
2661           byte += byte + (r + g + g + b + dith > 1020);
2662           if ((x & 7) == 7)
2663             {
2664               obptr[0] = byte;
2665               obptr++;
2666             }
2667         }
2668       if (x & 7)
2669         obptr[0] = byte << (8 - (x & 7));
2670       bptr += rowstride;
2671       obuf += bpl;
2672     }
2673 }
2674
2675 /* Returns a pointer to the stage buffer. */
2676 static guchar *
2677 gdk_rgb_ensure_stage (GdkRgbInfo *image_info)
2678 {
2679   if (image_info->stage_buf == NULL)
2680     image_info->stage_buf = g_malloc (GDK_SCRATCH_IMAGE_HEIGHT * STAGE_ROWSTRIDE);
2681   return image_info->stage_buf;
2682 }
2683
2684 /* This is slow. Speed me up, please. */
2685 static void
2686 gdk_rgb_32_to_stage (GdkRgbInfo *image_info,
2687                      guchar *buf, gint rowstride, gint width, gint height)
2688 {
2689   gint x, y;
2690   guchar *pi_start, *po_start;
2691   guchar *pi, *po;
2692
2693   pi_start = buf;
2694   po_start = gdk_rgb_ensure_stage (image_info);
2695   for (y = 0; y < height; y++)
2696     {
2697       pi = pi_start;
2698       po = po_start;
2699       for (x = 0; x < width; x++)
2700         {
2701           *po++ = *pi++;
2702           *po++ = *pi++;
2703           *po++ = *pi++;
2704           pi++;
2705         }
2706       pi_start += rowstride;
2707       po_start += STAGE_ROWSTRIDE;
2708     }
2709 }
2710
2711 /* Generic 32bit RGB conversion function - convert to 24bit packed, then
2712    go from there. */
2713 static void
2714 gdk_rgb_convert_32_generic (GdkRgbInfo *image_info, GdkImage *image,
2715                             gint x0, gint y0, gint width, gint height,
2716                             guchar *buf, gint rowstride,
2717                             gint x_align, gint y_align, GdkRgbCmap *cmap)
2718 {
2719   gdk_rgb_32_to_stage (image_info, buf, rowstride, width, height);
2720
2721   (*image_info->conv) (image_info, image, x0, y0, width, height,
2722                        image_info->stage_buf, STAGE_ROWSTRIDE,
2723                        x_align, y_align, cmap);
2724 }
2725
2726 /* Generic 32bit RGB conversion function - convert to 24bit packed, then
2727    go from there. */
2728 static void
2729 gdk_rgb_convert_32_generic_d (GdkRgbInfo *image_info, GdkImage *image,
2730                               gint x0, gint y0, gint width, gint height,
2731                               guchar *buf, gint rowstride,
2732                               gint x_align, gint y_align, GdkRgbCmap *cmap)
2733 {
2734   gdk_rgb_32_to_stage (image_info, buf, rowstride, width, height);
2735
2736   (*image_info->conv_d) (image_info, image, x0, y0, width, height,
2737                          image_info->stage_buf, STAGE_ROWSTRIDE,
2738                          x_align, y_align, cmap);
2739 }
2740
2741 /* This is slow. Speed me up, please. */
2742 static void
2743 gdk_rgb_gray_to_stage (GdkRgbInfo *image_info,
2744                        guchar *buf, gint rowstride, gint width, gint height)
2745 {
2746   gint x, y;
2747   guchar *pi_start, *po_start;
2748   guchar *pi, *po;
2749   guchar gray;
2750
2751   pi_start = buf;
2752   po_start = gdk_rgb_ensure_stage (image_info);
2753   for (y = 0; y < height; y++)
2754     {
2755       pi = pi_start;
2756       po = po_start;
2757       for (x = 0; x < width; x++)
2758         {
2759           gray = *pi++;
2760           *po++ = gray;
2761           *po++ = gray;
2762           *po++ = gray;
2763         }
2764       pi_start += rowstride;
2765       po_start += STAGE_ROWSTRIDE;
2766     }
2767 }
2768
2769 /* Generic gray conversion function - convert to 24bit packed, then go
2770    from there. */
2771 static void
2772 gdk_rgb_convert_gray_generic (GdkRgbInfo *image_info, GdkImage *image,
2773                               gint x0, gint y0, gint width, gint height,
2774                               guchar *buf, gint rowstride,
2775                               gint x_align, gint y_align, GdkRgbCmap *cmap)
2776 {
2777   gdk_rgb_gray_to_stage (image_info, buf, rowstride, width, height);
2778
2779   (*image_info->conv) (image_info, image, x0, y0, width, height,
2780                        image_info->stage_buf, STAGE_ROWSTRIDE,
2781                        x_align, y_align, cmap);
2782 }
2783
2784 static void
2785 gdk_rgb_convert_gray_generic_d (GdkRgbInfo *image_info, GdkImage *image,
2786                                 gint x0, gint y0, gint width, gint height,
2787                                 guchar *buf, gint rowstride,
2788                                 gint x_align, gint y_align, GdkRgbCmap *cmap)
2789 {
2790   gdk_rgb_gray_to_stage (image_info, buf, rowstride, width, height);
2791
2792   (*image_info->conv_d) (image_info, image, x0, y0, width, height,
2793                          image_info->stage_buf, STAGE_ROWSTRIDE,
2794                          x_align, y_align, cmap);
2795 }
2796
2797 /* Render grayscale using indexed method. */
2798 static void
2799 gdk_rgb_convert_gray_cmap (GdkRgbInfo *image_info, GdkImage *image,
2800                            gint x0, gint y0, gint width, gint height,
2801                            guchar *buf, gint rowstride,
2802                            gint x_align, gint y_align, GdkRgbCmap *cmap)
2803 {
2804   (*image_info->conv_indexed) (image_info, image, x0, y0, width, height,
2805                                buf, rowstride,
2806                                x_align, y_align, image_info->gray_cmap);
2807 }
2808
2809 #if 0
2810 static void
2811 gdk_rgb_convert_gray_cmap_d (GdkRgbInfo *image_info, GdkImage *image,
2812                              gint x0, gint y0, gint width, gint height,
2813                              guchar *buf, gint rowstride,
2814                              gint x_align, gint y_align, GdkRgbCmap *cmap)
2815 {
2816   (*image_info->conv_indexed_d) (image_info, image, x0, y0, width, height,
2817                                  buf, rowstride,
2818                                  x_align, y_align, image_info->gray_cmap);
2819 }
2820 #endif
2821
2822 /* This is slow. Speed me up, please. */
2823 static void
2824 gdk_rgb_indexed_to_stage (GdkRgbInfo *image_info,
2825                           guchar *buf, gint rowstride, gint width, gint height,
2826                           GdkRgbCmap *cmap)
2827 {
2828   gint x, y;
2829   guchar *pi_start, *po_start;
2830   guchar *pi, *po;
2831   gint rgb;
2832
2833   pi_start = buf;
2834   po_start = gdk_rgb_ensure_stage (image_info);
2835   for (y = 0; y < height; y++)
2836     {
2837       pi = pi_start;
2838       po = po_start;
2839       for (x = 0; x < width; x++)
2840         {
2841           rgb = cmap->colors[*pi++];
2842           *po++ = rgb >> 16;
2843           *po++ = (rgb >> 8) & 0xff;
2844           *po++ = rgb & 0xff;
2845         }
2846       pi_start += rowstride;
2847       po_start += STAGE_ROWSTRIDE;
2848     }
2849 }
2850
2851 /* Generic gray conversion function - convert to 24bit packed, then go
2852    from there. */
2853 static void
2854 gdk_rgb_convert_indexed_generic (GdkRgbInfo *image_info, GdkImage *image,
2855                                  gint x0, gint y0, gint width, gint height,
2856                                  guchar *buf, gint rowstride,
2857                                  gint x_align, gint y_align, GdkRgbCmap *cmap)
2858 {
2859   gdk_rgb_indexed_to_stage (image_info, buf, rowstride, width, height, cmap);
2860
2861   (*image_info->conv) (image_info, image, x0, y0, width, height,
2862                        image_info->stage_buf, STAGE_ROWSTRIDE,
2863                        x_align, y_align, cmap);
2864 }
2865
2866 static void
2867 gdk_rgb_convert_indexed_generic_d (GdkRgbInfo *image_info, GdkImage *image,
2868                                    gint x0, gint y0, gint width, gint height,
2869                                    guchar *buf, gint rowstride,
2870                                    gint x_align, gint y_align,
2871                                    GdkRgbCmap *cmap)
2872 {
2873   gdk_rgb_indexed_to_stage (image_info, buf, rowstride, width, height, cmap);
2874
2875   (*image_info->conv_d) (image_info, image, x0, y0, width, height,
2876                          image_info->stage_buf, STAGE_ROWSTRIDE,
2877                          x_align, y_align, cmap);
2878 }
2879
2880 /* Select a conversion function based on the visual and a
2881    representative image. */
2882 static void
2883 gdk_rgb_select_conv (GdkRgbInfo *image_info)
2884 {
2885   GdkByteOrder byte_order;
2886   gint depth, bpp, byterev;
2887   GdkVisualType vtype;
2888   guint32 red_mask, green_mask, blue_mask;
2889   GdkRgbConvFunc conv, conv_d;
2890   GdkRgbConvFunc conv_32, conv_32_d;
2891   GdkRgbConvFunc conv_gray, conv_gray_d;
2892   GdkRgbConvFunc conv_indexed, conv_indexed_d;
2893   gboolean mask_rgb, mask_bgr;
2894   GdkScreen *screen = gdk_visual_get_screen (image_info->visual);
2895
2896   depth = image_info->visual->depth;
2897
2898   bpp = _gdk_windowing_get_bits_for_depth (gdk_screen_get_display (screen),
2899                                            image_info->visual->depth);
2900
2901   byte_order = image_info->visual->byte_order;
2902   if (gdk_rgb_verbose)
2903     g_print ("Chose visual type=%s depth=%d, image bpp=%d, %s first\n",
2904              visual_names[image_info->visual->type], image_info->visual->depth,
2905              bpp, byte_order == GDK_LSB_FIRST ? "lsb" : "msb");
2906
2907 #if G_BYTE_ORDER == G_BIG_ENDIAN
2908   byterev = (byte_order == GDK_LSB_FIRST);
2909 #else
2910   byterev = (byte_order == GDK_MSB_FIRST);
2911 #endif
2912
2913   vtype = image_info->visual->type;
2914   if (vtype == GDK_VISUAL_DIRECT_COLOR)
2915     vtype = GDK_VISUAL_TRUE_COLOR;
2916
2917   red_mask = image_info->visual->red_mask;
2918   green_mask = image_info->visual->green_mask;
2919   blue_mask = image_info->visual->blue_mask;
2920
2921   mask_rgb = red_mask == 0xff0000 && green_mask == 0xff00 && blue_mask == 0xff;
2922   mask_bgr = red_mask == 0xff && green_mask == 0xff00 && blue_mask == 0xff0000;
2923
2924   conv = NULL;
2925   conv_d = NULL;
2926
2927   conv_32 = gdk_rgb_convert_32_generic;
2928   conv_32_d = gdk_rgb_convert_32_generic_d;
2929
2930   conv_gray = gdk_rgb_convert_gray_generic;
2931   conv_gray_d = gdk_rgb_convert_gray_generic_d;
2932
2933   conv_indexed = gdk_rgb_convert_indexed_generic;
2934   conv_indexed_d = gdk_rgb_convert_indexed_generic_d;
2935
2936   image_info->dith_default = FALSE;
2937
2938   if (image_info->bitmap)
2939     conv = gdk_rgb_convert_1;
2940   else if (bpp == 16 && depth == 16 && !byterev &&
2941       red_mask == 0xf800 && green_mask == 0x7e0 && blue_mask == 0x1f)
2942     {
2943       conv = gdk_rgb_convert_565;
2944       conv_d = gdk_rgb_convert_565_d;
2945       conv_gray = gdk_rgb_convert_565_gray;
2946       gdk_rgb_preprocess_dm_565 ();
2947     }
2948   else if (bpp == 16 && depth == 16 &&
2949            vtype == GDK_VISUAL_TRUE_COLOR && byterev &&
2950       red_mask == 0xf800 && green_mask == 0x7e0 && blue_mask == 0x1f)
2951     conv = gdk_rgb_convert_565_br;
2952
2953   else if (bpp == 16 && depth == 15 &&
2954            vtype == GDK_VISUAL_TRUE_COLOR && !byterev &&
2955       red_mask == 0x7c00 && green_mask == 0x3e0 && blue_mask == 0x1f)
2956     conv = gdk_rgb_convert_555;
2957
2958   else if (bpp == 16 && depth == 15 &&
2959            vtype == GDK_VISUAL_TRUE_COLOR && byterev &&
2960       red_mask == 0x7c00 && green_mask == 0x3e0 && blue_mask == 0x1f)
2961     conv = gdk_rgb_convert_555_br;
2962
2963   /* I'm not 100% sure about the 24bpp tests - but testing will show*/
2964   else if (bpp == 24 && depth == 24 && vtype == GDK_VISUAL_TRUE_COLOR &&
2965            ((mask_rgb && byte_order == GDK_LSB_FIRST) ||
2966             (mask_bgr && byte_order == GDK_MSB_FIRST)))
2967     conv = gdk_rgb_convert_888_lsb;
2968   else if (bpp == 24 && depth == 24 && vtype == GDK_VISUAL_TRUE_COLOR &&
2969            ((mask_rgb && byte_order == GDK_MSB_FIRST) ||
2970             (mask_bgr && byte_order == GDK_LSB_FIRST)))
2971     conv = gdk_rgb_convert_888_msb;
2972 #if G_BYTE_ORDER == G_BIG_ENDIAN
2973   else if (bpp == 32 && depth == 24 && vtype == GDK_VISUAL_TRUE_COLOR &&
2974            (mask_rgb && byte_order == GDK_LSB_FIRST))
2975     conv = gdk_rgb_convert_0888_br;
2976   else if (bpp == 32 && depth == 24 && vtype == GDK_VISUAL_TRUE_COLOR &&
2977            (mask_rgb && byte_order == GDK_MSB_FIRST))
2978     conv = gdk_rgb_convert_0888;
2979   else if (bpp == 32 && depth == 24 && vtype == GDK_VISUAL_TRUE_COLOR &&
2980            (mask_bgr && byte_order == GDK_MSB_FIRST))
2981     conv = gdk_rgb_convert_8880_br;
2982 #else
2983   else if (bpp == 32 && depth == 24 && vtype == GDK_VISUAL_TRUE_COLOR &&
2984            (mask_rgb && byte_order == GDK_MSB_FIRST))
2985     conv = gdk_rgb_convert_0888_br;
2986   else if (bpp == 32 && depth == 24 && vtype == GDK_VISUAL_TRUE_COLOR &&
2987            (mask_rgb && byte_order == GDK_LSB_FIRST))
2988     conv = gdk_rgb_convert_0888;
2989   else if (bpp == 32 && depth == 24 && vtype == GDK_VISUAL_TRUE_COLOR &&
2990            (mask_bgr && byte_order == GDK_LSB_FIRST))
2991     conv = gdk_rgb_convert_8880_br;
2992 #endif
2993
2994   else if (vtype == GDK_VISUAL_TRUE_COLOR && byte_order == GDK_LSB_FIRST)
2995     {
2996       conv = gdk_rgb_convert_truecolor_lsb;
2997       conv_d = gdk_rgb_convert_truecolor_lsb_d;
2998     }
2999   else if (vtype == GDK_VISUAL_TRUE_COLOR && byte_order == GDK_MSB_FIRST)
3000     {
3001       conv = gdk_rgb_convert_truecolor_msb;
3002       conv_d = gdk_rgb_convert_truecolor_msb_d;
3003     }
3004   else if (bpp == 8 &&
3005            depth <= 8 &&
3006            depth > 4 &&
3007            (vtype == GDK_VISUAL_PSEUDO_COLOR
3008             || vtype == GDK_VISUAL_STATIC_COLOR
3009 #ifdef ENABLE_GRAYSCALE
3010             || vtype == GDK_VISUAL_GRAYSCALE
3011 #endif
3012             ))
3013     {
3014       /* Mainly for Win32: use these conversion functions also for the
3015        * explicitly (on user request) restricted palette versions of
3016        * 256-color, i.e. depths 5, 6 and 7. On X11, such depths
3017        * probably never occur.
3018        */
3019       image_info->dith_default = TRUE;
3020       conv = gdk_rgb_convert_8;
3021       if (vtype != GDK_VISUAL_GRAYSCALE)
3022         {
3023           if (image_info->nred_shades == 6 &&
3024               image_info->ngreen_shades == 6 &&
3025               image_info->nblue_shades == 6)
3026             conv_d = gdk_rgb_convert_8_d666;
3027           else
3028             conv_d = gdk_rgb_convert_8_d;
3029         }
3030       conv_indexed = gdk_rgb_convert_8_indexed;
3031       conv_gray = gdk_rgb_convert_gray_cmap;
3032     }
3033   else if (bpp == 8 && depth == 8 && (vtype == GDK_VISUAL_STATIC_GRAY
3034 #ifdef not_ENABLE_GRAYSCALE
3035                                       || vtype == GDK_VISUAL_GRAYSCALE
3036 #endif
3037                                       ))
3038     {
3039       conv = gdk_rgb_convert_gray8;
3040       conv_gray = gdk_rgb_convert_gray8_gray;
3041     }
3042   else if (bpp == 8 && depth < 8 && depth >= 2 &&
3043            (vtype == GDK_VISUAL_STATIC_GRAY
3044             || vtype == GDK_VISUAL_GRAYSCALE))
3045     {
3046       conv = gdk_rgb_convert_gray4;
3047       conv_d = gdk_rgb_convert_gray4_d;
3048     }
3049   else if (bpp == 8 && depth < 8 && depth >= 3)
3050     {
3051       conv = gdk_rgb_convert_4;
3052     }
3053   else if (bpp == 4 && depth <= 4 && depth >= 2 &&
3054            (vtype == GDK_VISUAL_STATIC_GRAY
3055             || vtype == GDK_VISUAL_GRAYSCALE))
3056     {
3057       conv = gdk_rgb_convert_gray4_pack;
3058       conv_d = gdk_rgb_convert_gray4_d_pack;
3059     }
3060   else if (bpp == 4 && depth == 4 &&
3061            vtype == GDK_VISUAL_STATIC_COLOR)
3062     conv = gdk_rgb_convert_4_pack;
3063
3064   if (!conv)
3065     {
3066       g_warning ("Visual type=%s depth=%d, image bpp=%d, %s first\n"
3067                  "is not supported by GdkRGB. Please submit a bug report\n"
3068                  "with the above values to bugzilla.gnome.org",
3069                  visual_names[vtype], depth, bpp,
3070                  byte_order == GDK_LSB_FIRST ? "lsb" : "msb");
3071       exit (1);
3072     }
3073   
3074   if (conv_d == NULL)
3075     conv_d = conv;
3076
3077   image_info->conv = conv;
3078   image_info->conv_d = conv_d;
3079
3080   image_info->conv_32 = conv_32;
3081   image_info->conv_32_d = conv_32_d;
3082
3083   image_info->conv_gray = conv_gray;
3084   image_info->conv_gray_d = conv_gray_d;
3085
3086   image_info->conv_indexed = conv_indexed;
3087   image_info->conv_indexed_d = conv_indexed_d;
3088 }
3089
3090 static void
3091 gdk_draw_rgb_image_core (GdkRgbInfo *image_info,
3092                          GdkDrawable *drawable,
3093                          GdkGC *gc,
3094                          gint x,
3095                          gint y,
3096                          gint width,
3097                          gint height,
3098                          guchar *buf,
3099                          gint pixstride,
3100                          gint rowstride,
3101                          GdkRgbConvFunc conv,
3102                          GdkRgbCmap *cmap,
3103                          gint xdith,
3104                          gint ydith)
3105 {
3106   gint y0, x0;
3107   gint xs0, ys0;
3108   GdkImage *image;
3109   gint width1, height1;
3110   guchar *buf_ptr;
3111
3112   if (image_info->bitmap)
3113     {
3114       if (image_info->own_gc == NULL)
3115         image_info->own_gc = gdk_gc_new (drawable);
3116       gc = image_info->own_gc;
3117     }
3118   for (y0 = 0; y0 < height; y0 += GDK_SCRATCH_IMAGE_HEIGHT)
3119     {
3120       height1 = MIN (height - y0, GDK_SCRATCH_IMAGE_HEIGHT);
3121       for (x0 = 0; x0 < width; x0 += GDK_SCRATCH_IMAGE_WIDTH)
3122         {
3123           width1 = MIN (width - x0, GDK_SCRATCH_IMAGE_WIDTH);
3124           buf_ptr = buf + y0 * rowstride + x0 * pixstride;
3125
3126           image = _gdk_image_get_scratch (gdk_drawable_get_screen (drawable), 
3127                                           width1, height1,
3128                                           image_info->visual->depth, &xs0, &ys0);
3129
3130           conv (image_info, image, xs0, ys0, width1, height1, buf_ptr, rowstride,
3131                 x + x0 + xdith, y + y0 + ydith, cmap);
3132
3133 #ifndef DONT_ACTUALLY_DRAW
3134           gdk_draw_image (drawable, gc,
3135                           image, xs0, ys0, x + x0, y + y0, width1, height1);
3136 #endif
3137         }
3138     }
3139 }
3140
3141 static GdkRgbInfo *
3142 gdk_rgb_get_info_from_drawable (GdkDrawable *drawable)
3143 {
3144   GdkColormap *cmap = gdk_drawable_get_colormap (drawable);
3145   GdkScreen *screen = gdk_drawable_get_screen (drawable);
3146
3147   if (!cmap)
3148     {
3149       /* This guessing is required to maintain backward compatibility,
3150        * but would otherwise be a bad thing
3151        */
3152
3153       gint depth = gdk_drawable_get_depth (drawable);
3154       GdkColormap *rgb_cmap = gdk_screen_get_rgb_colormap (screen);
3155       if (depth == gdk_colormap_get_visual (rgb_cmap)->depth)
3156         cmap = rgb_cmap;
3157       else
3158         {
3159           g_warning ("The gdk_draw_*_image require the drawable argument to\n"
3160                      "have a specified colormap. All windows have a colormap,\n"
3161                      "however, pixmaps only have colormap by default if they\n"
3162                      "were created with a non-NULL window argument. Otherwise\n"
3163                      "a colormap must be set on them with gdk_drawable_set_colormap");
3164           return NULL;
3165         }
3166     }
3167
3168   return gdk_rgb_get_info_from_colormap (cmap);
3169 }
3170
3171 void
3172 gdk_draw_rgb_image (GdkDrawable *drawable,
3173                     GdkGC *gc,
3174                     gint x,
3175                     gint y,
3176                     gint width,
3177                     gint height,
3178                     GdkRgbDither dith,
3179                     guchar *rgb_buf,
3180                     gint rowstride)
3181 {
3182   GdkRgbInfo *image_info = gdk_rgb_get_info_from_drawable (drawable);
3183   if (!image_info)
3184     return;
3185   
3186   if (dith == GDK_RGB_DITHER_NONE || (dith == GDK_RGB_DITHER_NORMAL &&
3187                                       !image_info->dith_default))
3188     gdk_draw_rgb_image_core (image_info, drawable, gc, x, y, width, height,
3189                              rgb_buf, 3, rowstride, image_info->conv, NULL,
3190                              0, 0);
3191   else
3192     gdk_draw_rgb_image_core (image_info, drawable, gc, x, y, width, height,
3193                              rgb_buf, 3, rowstride, image_info->conv_d, NULL,
3194                              0, 0);
3195 }
3196
3197 void
3198 gdk_draw_rgb_image_dithalign (GdkDrawable *drawable,
3199                               GdkGC *gc,
3200                               gint x,
3201                               gint y,
3202                               gint width,
3203                               gint height,
3204                               GdkRgbDither dith,
3205                               guchar *rgb_buf,
3206                               gint rowstride,
3207                               gint xdith,
3208                               gint ydith)
3209 {
3210   GdkRgbInfo *image_info = gdk_rgb_get_info_from_drawable (drawable);
3211   if (!image_info)
3212     return;
3213   
3214   if (dith == GDK_RGB_DITHER_NONE || (dith == GDK_RGB_DITHER_NORMAL &&
3215                                       !image_info->dith_default))
3216     gdk_draw_rgb_image_core (image_info, drawable, gc, x, y, width, height,
3217                              rgb_buf, 3, rowstride, image_info->conv, NULL,
3218                              xdith, ydith);
3219   else
3220     gdk_draw_rgb_image_core (image_info, drawable, gc, x, y, width, height,
3221                              rgb_buf, 3, rowstride, image_info->conv_d, NULL,
3222                              xdith, ydith);
3223 }
3224
3225 void
3226 gdk_draw_rgb_32_image (GdkDrawable *drawable,
3227                        GdkGC *gc,
3228                        gint x,
3229                        gint y,
3230                        gint width,
3231                        gint height,
3232                        GdkRgbDither dith,
3233                        guchar *buf,
3234                        gint rowstride)
3235 {
3236   GdkRgbInfo *image_info = gdk_rgb_get_info_from_drawable (drawable);
3237   if (!image_info)
3238     return;
3239   
3240   if (dith == GDK_RGB_DITHER_NONE || (dith == GDK_RGB_DITHER_NORMAL &&
3241                                       !image_info->dith_default))
3242     gdk_draw_rgb_image_core (image_info, drawable, gc, x, y, width, height,
3243                              buf, 4, rowstride,
3244                              image_info->conv_32, NULL, 0, 0);
3245   else
3246     gdk_draw_rgb_image_core (image_info, drawable, gc, x, y, width, height,
3247                              buf, 4, rowstride,
3248                              image_info->conv_32_d, NULL, 0, 0);
3249 }
3250
3251 /**
3252  * gdk_draw_rgb_32_image_dithalign:
3253  * @drawable: a #GdkDrawable
3254  * @gc: a #GdkGC
3255  * @x: X coordinate on @drawable where image should go
3256  * @y: Y coordinate on @drawable where image should go
3257  * @width: width of area of image to draw
3258  * @height: height of area of image to draw
3259  * @dith: dithering mode
3260  * @buf: RGB image data
3261  * @rowstride: rowstride of RGB image data
3262  * @xdith: X dither offset
3263  * @ydith: Y dither offset
3264  *
3265  * Like gdk_draw_rgb_32_image(), but allows you to specify the dither
3266  * offsets. See gdk_draw_rgb_image_dithalign() for more details.
3267  * 
3268  **/
3269 void
3270 gdk_draw_rgb_32_image_dithalign (GdkDrawable *drawable,
3271                                  GdkGC *gc,
3272                                  gint x,
3273                                  gint y,
3274                                  gint width,
3275                                  gint height,
3276                                  GdkRgbDither dith,
3277                                  guchar *buf,
3278                                  gint rowstride,
3279                                  gint xdith,
3280                                  gint ydith)
3281 {
3282   GdkRgbInfo *image_info = gdk_rgb_get_info_from_drawable (drawable);
3283   if (!image_info)
3284     return;
3285   
3286   if (dith == GDK_RGB_DITHER_NONE || (dith == GDK_RGB_DITHER_NORMAL &&
3287                                       !image_info->dith_default))
3288     gdk_draw_rgb_image_core (image_info, drawable, gc, x, y, width, height,
3289                              buf, 4, rowstride,
3290                              image_info->conv_32, NULL, xdith, ydith);
3291   else
3292     gdk_draw_rgb_image_core (image_info, drawable, gc, x, y, width, height,
3293                              buf, 4, rowstride,
3294                              image_info->conv_32_d, NULL, xdith, ydith);
3295 }
3296
3297 static void
3298 gdk_rgb_make_gray_cmap (GdkRgbInfo *info)
3299 {
3300   guint32 rgb[256];
3301   gint i;
3302
3303   for (i = 0; i < 256; i++)
3304     rgb[i] = (i << 16)  | (i << 8) | i;
3305   info->gray_cmap = gdk_rgb_cmap_new (rgb, 256);
3306 }
3307
3308 void
3309 gdk_draw_gray_image (GdkDrawable *drawable,
3310                      GdkGC *gc,
3311                      gint x,
3312                      gint y,
3313                      gint width,
3314                      gint height,
3315                      GdkRgbDither dith,
3316                      guchar *buf,
3317                      gint rowstride)
3318 {
3319   GdkRgbInfo *image_info = gdk_rgb_get_info_from_drawable (drawable);
3320   if (!image_info)
3321     return;
3322   
3323   if (image_info->bpp == 1 &&
3324       image_info->gray_cmap == NULL &&
3325       (image_info->visual->type == GDK_VISUAL_PSEUDO_COLOR ||
3326        image_info->visual->type == GDK_VISUAL_STATIC_COLOR ||
3327        image_info->visual->type == GDK_VISUAL_GRAYSCALE))
3328     gdk_rgb_make_gray_cmap (image_info);
3329   
3330   if (dith == GDK_RGB_DITHER_NONE || (dith == GDK_RGB_DITHER_NORMAL &&
3331                                       !image_info->dith_default))
3332     gdk_draw_rgb_image_core (image_info, drawable, gc, x, y, width, height,
3333                              buf, 1, rowstride,
3334                              image_info->conv_gray, NULL, 0, 0);
3335   else
3336     gdk_draw_rgb_image_core (image_info, drawable, gc, x, y, width, height,
3337                              buf, 1, rowstride,
3338                              image_info->conv_gray_d, NULL, 0, 0);
3339 }
3340
3341 static GdkRgbCmapInfo *
3342 gdk_rgb_cmap_get_info (GdkRgbCmap *cmap,
3343                        GdkRgbInfo *image_info)
3344 {
3345   GSList *tmp_list;
3346   GdkRgbCmapInfo *cmap_info;
3347   int i, j;
3348   guint32 rgb;
3349
3350   /* We don't need a LUT for TrueColor or DirectColor visuals */
3351   if (image_info->bpp != 1 ||
3352       !(image_info->visual->type == GDK_VISUAL_PSEUDO_COLOR ||
3353         image_info->visual->type == GDK_VISUAL_STATIC_COLOR ||
3354         image_info->visual->type == GDK_VISUAL_GRAYSCALE))
3355     return NULL;
3356   
3357   tmp_list = cmap->info_list;
3358
3359   while (tmp_list)
3360     {
3361      cmap_info = tmp_list->data;
3362       if (cmap_info->image_info == image_info)
3363         return cmap_info;
3364     }
3365
3366   cmap_info = g_new (GdkRgbCmapInfo, 1);
3367   cmap_info->image_info = image_info;
3368   cmap_info->cmap = cmap;
3369   
3370   for (i = 0; i < cmap->n_colors; i++)
3371     {
3372       rgb = cmap->colors[i];
3373       j = ((rgb & 0xf00000) >> 12) |
3374         ((rgb & 0xf000) >> 8) |
3375         ((rgb & 0xf0) >> 4);
3376 #ifdef VERBOSE
3377       g_print ("%d %x %x\n", i, j, image_info->colorcube[j]);
3378 #endif
3379       cmap_info->lut[i] = image_info->colorcube[j];
3380     }
3381
3382   cmap->info_list = g_slist_prepend (cmap->info_list, cmap_info);
3383   image_info->cmap_info_list = g_slist_prepend (image_info->cmap_info_list, cmap_info);
3384   
3385   return cmap_info;
3386 }
3387
3388 GdkRgbCmap *
3389 gdk_rgb_cmap_new (guint32 *colors, gint n_colors)
3390 {
3391   GdkRgbCmap *cmap;
3392     
3393   g_return_val_if_fail (n_colors >= 0, NULL);
3394   g_return_val_if_fail (n_colors <= 256, NULL);
3395
3396   cmap = g_new (GdkRgbCmap, 1);
3397
3398   cmap->n_colors = n_colors;
3399   memcpy (cmap->colors, colors, n_colors * sizeof(guint32));
3400
3401   cmap->info_list = NULL;
3402   
3403   return cmap;
3404 }
3405
3406 void
3407 gdk_rgb_cmap_free (GdkRgbCmap *cmap)
3408 {
3409   GSList *tmp_list;
3410
3411   tmp_list = cmap->info_list;
3412   while (tmp_list)
3413     {
3414       GdkRgbCmapInfo *cmap_info = tmp_list->data;
3415       cmap_info->image_info->cmap_info_list = g_slist_remove (cmap_info->image_info->cmap_info_list, cmap_info);
3416       g_free (cmap_info);
3417       tmp_list = tmp_list->next;
3418     }
3419   g_slist_free (cmap->info_list);
3420   
3421   g_free (cmap);
3422 }
3423
3424 void
3425 gdk_draw_indexed_image (GdkDrawable *drawable,
3426                         GdkGC *gc,
3427                         gint x,
3428                         gint y,
3429                         gint width,
3430                         gint height,
3431                         GdkRgbDither dith,
3432                         guchar *buf,
3433                         gint rowstride,
3434                         GdkRgbCmap *cmap)
3435 {
3436   GdkRgbInfo *image_info = gdk_rgb_get_info_from_drawable (drawable);
3437   if (!image_info)
3438     return;
3439   
3440   if (dith == GDK_RGB_DITHER_NONE || (dith == GDK_RGB_DITHER_NORMAL &&
3441                                       !image_info->dith_default))
3442     gdk_draw_rgb_image_core (image_info, drawable, gc, x, y, width, height,
3443                              buf, 1, rowstride,
3444                              image_info->conv_indexed, cmap, 0, 0);
3445   else
3446     gdk_draw_rgb_image_core (image_info, drawable, gc, x, y, width, height,
3447                              buf, 1, rowstride,
3448                              image_info->conv_indexed_d, cmap, 0, 0);
3449 }
3450
3451 gboolean
3452 gdk_rgb_colormap_ditherable (GdkColormap *cmap)
3453 {
3454   GdkRgbInfo *image_info = gdk_rgb_get_info_from_colormap (cmap);
3455
3456   return (image_info->conv != image_info->conv_d);
3457 }
3458
3459 gboolean
3460 gdk_rgb_ditherable (void)
3461 {
3462   return gdk_rgb_colormap_ditherable (gdk_rgb_get_colormap ());
3463 }
3464
3465 /**
3466  * gdk_rgb_get_colormap:
3467  * 
3468  * Get the preferred colormap for rendering image data.  Not a
3469  * very useful function; historically, GDK could only render RGB image
3470  * data to one colormap and visual, but in the current version it can
3471  * render to any colormap and visual. So there's no need to call this
3472  * function.
3473  * 
3474  * Return value: the preferred colormap
3475  **/
3476 GdkColormap *
3477 gdk_rgb_get_colormap (void)
3478 {
3479   static GdkColormap *cmap = NULL;
3480   if (!cmap)
3481     {
3482       GdkRgbInfo *image_info = gdk_rgb_create_info (gdk_rgb_choose_visual (gdk_screen_get_default ()), NULL);
3483       cmap = image_info->cmap;
3484     }
3485
3486   return cmap;
3487 }
3488
3489 /**
3490  * gdk_screen_get_rgb_colormap:
3491  * @screen : a #GdkScreen.
3492  * 
3493  * Gets the preferred colormap for rendering image data on @screen.
3494  * Not a very useful function; historically, GDK could only render RGB
3495  * image data to one colormap and visual, but in the current version
3496  * it can render to any colormap and visual. So there's no need to
3497  * call this function.
3498  * 
3499  * Return value: the preferred colormap
3500  **/
3501 GdkColormap *
3502 gdk_screen_get_rgb_colormap (GdkScreen *screen)
3503 {
3504   GdkColormap *cmap;
3505   g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
3506   cmap = g_object_get_data (G_OBJECT (screen), "rgb-colormap"); 
3507   if (!cmap)
3508     {
3509       GdkRgbInfo *image_info = gdk_rgb_create_info (gdk_rgb_choose_visual (screen), NULL);
3510       cmap = image_info->cmap;
3511       g_object_set_data (G_OBJECT (screen), "rgb-colormap", cmap);
3512     }
3513
3514   return cmap;
3515 }
3516
3517 /**
3518  * gdk_screen_get_rgb_visual:
3519  * @screen : a #GdkScreen
3520  * 
3521  * Gets a "preferred visual" chosen by GdkRGB for rendering image data
3522  * on @screen. In previous versions of
3523  * GDK, this was the only visual GdkRGB could use for rendering. In
3524  * current versions, it's simply the visual GdkRGB would have chosen as 
3525  * the optimal one in those previous versions. GdkRGB can now render to 
3526  * drawables with any visual.
3527  * 
3528  * Return value: The #GdkVisual chosen by GdkRGB.
3529  **/
3530 GdkVisual *
3531 gdk_screen_get_rgb_visual (GdkScreen *screen)
3532 {
3533   g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
3534   return gdk_colormap_get_visual (gdk_screen_get_rgb_colormap (screen));
3535 }
3536
3537 /**
3538  * gdk_rgb_get_visual:
3539  * 
3540  * Gets a "preferred visual" chosen by GdkRGB for rendering image data
3541  * on the default screen. In previous versions of GDK, this was the
3542  * only visual GdkRGB could use for rendering. In current versions,
3543  * it's simply the visual GdkRGB would have chosen as the optimal one
3544  * in those previous versions. GdkRGB can now render to drawables with
3545  * any visual.
3546  * 
3547  * Return value: The #GdkVisual chosen by GdkRGB.
3548  **/
3549 GdkVisual *
3550 gdk_rgb_get_visual (void)
3551 {
3552   return gdk_screen_get_rgb_visual (gdk_screen_get_default ());
3553 }