]> Pileus Git - ~andy/gtk/blob - gdk/gdkrgb.c
1138aafb9c21d492cdf143dc1f4b75894d39e287
[~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 Library 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  * Library General Public License for more details.
13  *
14  * You should have received a copy of the GNU Library 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 #include <math.h>
26 #include <stdio.h>
27
28 #define ENABLE_GRAYSCALE
29
30 #ifdef GDK_RGB_STANDALONE
31
32 /* Compiling as a standalone module (i.e. with Gtk 1.0) */
33 /* gtk/gtk.h is already included in gdkrgbstub.c */
34 #include "config.h"
35 #include <gdk/gdkprivate.h>
36
37 #else
38
39 /* Compiling as a part of Gtk 1.1 or later */
40 #include "../config.h"
41 #include "gdk.h"
42 #include "gdkprivate.h"
43
44 #endif
45
46 #include "gdkrgb.h"
47
48 typedef struct _GdkRgbInfo   GdkRgbInfo;
49
50 typedef void (*GdkRgbConvFunc) (GdkImage *image,
51                                 gint x0, gint y0,
52                                 gint width, gint height,
53                                 guchar *buf, int rowstride,
54                                 gint x_align, gint y_align,
55                                 GdkRgbCmap *cmap);
56
57 /* Some of these fields should go, as they're not being used at all.
58    Globals should generally migrate into here - it's very likely that
59    we'll want to run more than one GdkRgbInfo context at the same time
60    (i.e. some but not all windows have privately installed
61    colormaps). */
62
63 struct _GdkRgbInfo
64 {
65   GdkVisual *visual;
66   GdkColormap *cmap;
67
68   gulong *color_pixels;
69   gulong *gray_pixels;
70   gulong *reserved_pixels;
71
72   guint nred_shades;
73   guint ngreen_shades;
74   guint nblue_shades;
75   guint ngray_shades;
76   guint nreserved;
77
78   guint bpp;
79   gint cmap_alloced;
80   gdouble gamma;
81
82   /* Generally, the stage buffer is used to convert 32bit RGB, gray,
83      and indexed images into 24 bit packed RGB. */
84   guchar *stage_buf;
85
86   GdkRgbCmap *gray_cmap;
87
88   gboolean dith_default;
89
90   /* Convert functions */
91   GdkRgbConvFunc conv;
92   GdkRgbConvFunc conv_d;
93
94   GdkRgbConvFunc conv_32;
95   GdkRgbConvFunc conv_32_d;
96
97   GdkRgbConvFunc conv_gray;
98   GdkRgbConvFunc conv_gray_d;
99
100   GdkRgbConvFunc conv_indexed;
101   GdkRgbConvFunc conv_indexed_d;
102 };
103
104 static gboolean gdk_rgb_install_cmap = FALSE;
105 static gint gdk_rgb_min_colors = 5 * 5 * 5;
106 static gboolean gdk_rgb_verbose = FALSE;
107
108 #define IMAGE_WIDTH 256
109 #define STAGE_ROWSTRIDE (IMAGE_WIDTH * 3)
110 #define IMAGE_HEIGHT 64
111 #define N_IMAGES 6
112
113 static GdkRgbInfo *image_info = NULL;
114 static GdkImage *static_image[N_IMAGES];
115 static gint static_image_idx;
116
117 static guchar *colorcube;
118 static guchar *colorcube_d;
119
120 static gint
121 gdk_rgb_cmap_fail (const char *msg, GdkColormap *cmap, gulong *pixels)
122 {
123   gulong free_pixels[256];
124   gint n_free;
125   gint i;
126
127 #ifdef VERBOSE
128   g_print ("%s", msg);
129 #endif
130   n_free = 0;
131   for (i = 0; i < 256; i++)
132     if (pixels[i] < 256)
133       free_pixels[n_free++] = pixels[i];
134   if (n_free)
135     gdk_colors_free (cmap, free_pixels, n_free, 0);
136   return 0;
137 }
138
139 static void
140 gdk_rgb_make_colorcube (gulong *pixels, gint nr, gint ng, gint nb)
141 {
142   guchar rt[16], gt[16], bt[16];
143   gint i;
144
145   colorcube = g_new (guchar, 4096);
146   for (i = 0; i < 16; i++)
147     {
148       rt[i] = ng * nb * ((i * 17 * (nr - 1) + 128) >> 8);
149       gt[i] = nb * ((i * 17 * (ng - 1) + 128) >> 8);
150       bt[i] = ((i * 17 * (nb - 1) + 128) >> 8);
151     }
152
153   for (i = 0; i < 4096; i++)
154     {
155       colorcube[i] = pixels[rt[i >> 8] + gt[(i >> 4) & 0x0f] + bt[i & 0x0f]];
156 #ifdef VERBOSE
157       g_print ("%03x %02x %x %x %x\n", i, colorcube[i], rt[i >> 8], gt[(i >> 4) & 0x0f], bt[i & 0x0f]);
158 #endif
159     }
160 }
161
162 /* this is the colorcube suitable for dithering */
163 static void
164 gdk_rgb_make_colorcube_d (gulong *pixels, gint nr, gint ng, gint nb)
165 {
166   gint r, g, b;
167   gint i;
168
169   colorcube_d = g_new (guchar, 512);
170   for (i = 0; i < 512; i++)
171     {
172       r = MIN (nr - 1, i >> 6);
173       g = MIN (ng - 1, (i >> 3) & 7);
174       b = MIN (nb - 1, i & 7);
175       colorcube_d[i] = pixels[(r * ng + g) * nb + b];
176     }
177 }
178
179 /* Try installing a color cube of the specified size.
180    Make the colorcube and return TRUE on success */
181 static gint
182 gdk_rgb_try_colormap (gint nr, gint ng, gint nb)
183 {
184   gint r, g, b;
185   gint ri, gi, bi;
186   gint r0, g0, b0;
187   GdkColormap *cmap;
188   GdkColor color;
189   gulong pixels[256];
190   gulong junk[256];
191   gint i;
192   gint d2;
193   gint colors_needed;
194   gint idx;
195   gint best[256];
196
197   if (nr * ng * nb < gdk_rgb_min_colors)
198     return FALSE;
199
200   if (image_info->cmap_alloced)
201     cmap = image_info->cmap;
202   else
203     cmap = gdk_colormap_get_system ();
204
205   colors_needed = nr * ng * nb;
206   for (i = 0; i < 256; i++)
207     {
208       best[i] = 192;
209       pixels[i] = 256;
210     }
211
212 #ifndef GAMMA
213   if (!gdk_rgb_install_cmap)
214   /* find color cube colors that are already present */
215   for (i = 0; i < MIN (256, cmap->size); i++)
216     {
217       r = cmap->colors[i].red >> 8;
218       g = cmap->colors[i].green >> 8;
219       b = cmap->colors[i].blue >> 8;
220       ri = (r * (nr - 1) + 128) >> 8;
221       gi = (g * (ng - 1) + 128) >> 8;
222       bi = (b * (nb - 1) + 128) >> 8;
223       r0 = ri * 255 / (nr - 1);
224       g0 = gi * 255 / (ng - 1);
225       b0 = bi * 255 / (nb - 1);
226       idx = ((ri * nr) + gi) * nb + bi;
227       d2 = (r - r0) * (r - r0) + (g - g0) * (g - g0) + (b - b0) * (b - b0);
228       if (d2 < best[idx]) {
229         if (pixels[idx] < 256)
230           gdk_colors_free (cmap, pixels + idx, 1, 0);
231         else
232           colors_needed--;
233         color = cmap->colors[i];
234         if (!gdk_color_alloc (cmap, &color))
235           return gdk_rgb_cmap_fail ("error allocating system color\n",
236                                     cmap, pixels);
237         pixels[idx] = color.pixel; /* which is almost certainly i */
238         best[idx] = d2;
239       }
240     }
241 #endif
242
243   if (colors_needed)
244     {
245       if (!gdk_colors_alloc (cmap, 0, NULL, 0, junk, colors_needed))
246         {
247           char tmp_str[80];
248           
249           sprintf (tmp_str,
250                    "%d %d %d colormap failed (in gdk_colors_alloc)\n",
251                    nr, ng, nb);
252           return gdk_rgb_cmap_fail (tmp_str, cmap, pixels);
253         }
254
255       gdk_colors_free (cmap, junk, colors_needed, 0);
256     }
257
258   for (r = 0, i = 0; r < nr; r++)
259     for (g = 0; g < ng; g++)
260       for (b = 0; b < nb; b++, i++)
261         {
262           if (pixels[i] == 256)
263             {
264               color.red = r * 65535 / (nr - 1);
265               color.green = g * 65535 / (ng - 1);
266               color.blue = b * 65535 / (nb - 1);
267
268 #ifdef GAMMA
269               color.red = 65535 * pow (color.red / 65535.0, 0.5);
270               color.green = 65535 * pow (color.green / 65535.0, 0.5);
271               color.blue = 65535 * pow (color.blue / 65535.0, 0.5);
272 #endif
273
274               if (!gdk_color_alloc (cmap, &color))
275                 {
276                   char tmp_str[80];
277
278                   sprintf (tmp_str, "%d %d %d colormap failed\n",
279                            nr, ng, nb);
280                   return gdk_rgb_cmap_fail (tmp_str,
281                                             cmap, pixels);
282                 }
283               pixels[i] = color.pixel;
284             }
285 #ifdef VERBOSE
286           g_print ("%d: %lx\n", i, pixels[i]);
287 #endif
288         }
289
290   image_info->nred_shades = nr;
291   image_info->ngreen_shades = ng;
292   image_info->nblue_shades = nb;
293   gdk_rgb_make_colorcube (pixels, nr, ng, nb);
294   gdk_rgb_make_colorcube_d (pixels, nr, ng, nb);
295   return TRUE;
296 }
297
298 /* Return TRUE on success. */
299 static gboolean
300 gdk_rgb_do_colormaps (void)
301 {
302   const gint sizes[][3] = {
303     /*    { 6, 7, 6 }, */
304     { 6, 6, 6 }, 
305     { 6, 6, 5 }, 
306     { 6, 6, 4 }, 
307     { 5, 5, 5 }, 
308     { 5, 5, 4 }, 
309     { 4, 4, 4 }, 
310     { 4, 4, 3 }, 
311     { 3, 3, 3 }, 
312     { 2, 2, 2 }
313   };
314   const gint n_sizes = sizeof(sizes) / (3 * sizeof(gint));
315   gint i;
316
317   for (i = 0; i < n_sizes; i++)
318     if (gdk_rgb_try_colormap (sizes[i][0], sizes[i][1], sizes[i][2]))
319       return TRUE;
320   return FALSE;
321 }
322
323 void
324 gdk_rgb_set_verbose (gboolean verbose)
325 {
326   gdk_rgb_verbose = verbose;
327 }
328
329 void
330 gdk_rgb_set_install (gboolean install)
331 {
332   gdk_rgb_install_cmap = install;
333 }
334
335 void
336 gdk_rgb_set_min_colors (gint min_colors)
337 {
338   gdk_rgb_min_colors = min_colors;
339 }
340
341 /* Return a "score" based on the following criteria (in hex):
342
343    x000 is the quality - 1 is 1bpp, 2 is 4bpp,
344                          4 is 8bpp,
345                          7 is 15bpp truecolor, 8 is 16bpp truecolor,
346                          9 is 24bpp truecolor.
347    0x00 is the speed - 1 is the normal case,
348                        2 means faster than normal
349    00x0 gets a point for being the system visual
350    000x gets a point for being pseudocolor
351
352    A caveat: in the 8bpp modes, being the system visual seems to be
353    quite important. Thus, all of the 8bpp modes should be ranked at
354    the same speed.
355 */
356 static guint32
357 gdk_rgb_score_visual (GdkVisual *visual)
358 {
359   guint32 quality, speed, sys, pseudo;
360   gchar* visual_names[] =
361   {
362     "static gray",
363     "grayscale",
364     "static color",
365     "pseudo color",
366     "true color",
367     "direct color",
368   };
369
370
371   quality = 0;
372   speed = 1;
373   sys = 0;
374   if (visual->type == GDK_VISUAL_TRUE_COLOR ||
375       visual->type == GDK_VISUAL_DIRECT_COLOR)
376     {
377       if (visual->depth == 24)
378         {
379           quality = 9;
380           /* Should test for MSB visual here, and set speed if so. */
381         }
382       else if (visual->depth == 16)
383         quality = 8;
384       else if (visual->depth == 15)
385         quality = 7;
386       else if (visual->depth == 8)
387         quality = 4;
388     }
389   else if (visual->type == GDK_VISUAL_PSEUDO_COLOR)
390     {
391       if (visual->depth == 8)
392         quality = 4;
393     }
394   else if (visual->type == GDK_VISUAL_STATIC_GRAY
395 #ifdef ENABLE_GRAYSCALE
396            || visual->type == GDK_VISUAL_GRAYSCALE
397 #endif
398            )
399     {
400       if (visual->depth == 8)
401         quality = 4;
402       else if (visual->depth == 4)
403         quality = 2;
404       else if (visual->depth == 1)
405         quality = 1;
406     }
407
408   if (quality == 0)
409     return 0;
410
411   sys = (visual == gdk_visual_get_system ());
412
413   pseudo = (visual->type == GDK_VISUAL_PSEUDO_COLOR);
414
415   if (gdk_rgb_verbose)
416     g_print ("Visual 0x%x, type = %s, depth = %d, %x:%x:%x%s; score=%x\n",
417              (gint)(((GdkVisualPrivate *)visual)->xvisual->visualid),
418              visual_names[visual->type],
419              visual->depth,
420              visual->red_mask,
421              visual->green_mask,
422              visual->blue_mask,
423              sys ? " (system)" : "",
424              (quality << 12) | (speed << 8) | (sys << 4) | pseudo);
425
426   return (quality << 12) | (speed << 8) | (sys << 4) | pseudo;
427 }
428
429 static void
430 gdk_rgb_choose_visual (void)
431 {
432   GList *visuals;
433   guint32 score, best_score;
434   GdkVisual *visual, *best_visual;
435
436   visuals = gdk_list_visuals ();
437
438   best_visual = visuals->data;
439   best_score = gdk_rgb_score_visual (best_visual);
440   visuals = visuals->next;
441   while (visuals)
442     {
443       visual = visuals->data;
444       score = gdk_rgb_score_visual (visual);
445       if (score > best_score)
446         {
447           best_score = score;
448           best_visual = visual;
449         }
450       visuals = visuals->next;
451     }
452
453   image_info->visual = best_visual;
454 }
455
456 static void gdk_rgb_select_conv (GdkImage *image);
457
458 static void
459 gdk_rgb_set_gray_cmap (GdkColormap *cmap)
460 {
461   gint i;
462   GdkColor color;
463   gint status;
464   gulong pixels[256];
465   gint r, g, b, gray;
466
467   for (i = 0; i < 256; i++)
468     {
469       color.pixel = i;
470       color.red = i * 257;
471       color.green = i * 257;
472       color.blue = i * 257;
473       status = gdk_color_alloc (cmap, &color);
474       pixels[i] = color.pixel;
475 #ifdef VERBOSE
476       g_print ("allocating pixel %d, %x %x %x, result %d\n",
477                color.pixel, color.red, color.green, color.blue, status);
478 #endif
479     }
480
481   /* Now, we make fake colorcubes - we ultimately just use the pseudocolor
482      methods. */
483
484   colorcube = g_new (guchar, 4096);
485
486   for (i = 0; i < 4096; i++)
487     {
488       r = (i >> 4) & 0xf0;
489       r = r | r >> 4;
490       g = i & 0xf0;
491       g = g | g >> 4;
492       b = (i << 4 & 0xf0);
493       b = b | b >> 4;
494       gray = (g + ((r + b) >> 1)) >> 1;
495       colorcube[i] = pixels[gray];
496     }
497 }
498
499 void
500 gdk_rgb_init (void)
501 {
502   gint i;
503   gint byte_order[1] = { 1 };
504
505   /* check endian sanity */
506 #ifdef WORDS_BIGENDIAN
507   if (((char *)byte_order)[0] == 1)
508     g_error ("gdk_rgb_init: WORDS_BIGENDIAN is defined, but this is a little endian machine.\n\n");
509 #else
510   if (((char *)byte_order)[0] != 1)
511     g_error ("gdk_rgb_init: WORDS_BIGENDIAN is not defined, but this is a little endian machine.\n\n");
512 #endif
513
514   if (image_info == NULL)
515     {
516       image_info = g_new0 (GdkRgbInfo, 1);
517
518       image_info->visual = NULL;
519       image_info->cmap = NULL;
520
521       image_info->color_pixels = NULL;
522       image_info->gray_pixels = NULL;
523       image_info->reserved_pixels = NULL;
524
525       image_info->nred_shades = 6;
526       image_info->ngreen_shades = 6;
527       image_info->nblue_shades = 4;
528       image_info->ngray_shades = 24;
529       image_info->nreserved = 0;
530
531       image_info->bpp = 0;
532       image_info->cmap_alloced = FALSE;
533       image_info->gamma = 1.0;
534
535       image_info->stage_buf = NULL;
536
537       gdk_rgb_choose_visual ();
538
539       if (image_info->visual->type == GDK_VISUAL_PSEUDO_COLOR)
540         {
541           if (gdk_rgb_install_cmap ||
542               image_info->visual != gdk_visual_get_system ())
543             {
544               image_info->cmap = gdk_colormap_new (image_info->visual, FALSE);
545               image_info->cmap_alloced = TRUE;
546             }
547           if (!gdk_rgb_do_colormaps ())
548             {
549               image_info->cmap = gdk_colormap_new (image_info->visual, FALSE);
550               image_info->cmap_alloced = TRUE;
551               gdk_rgb_do_colormaps ();
552             }
553           if (gdk_rgb_verbose)
554             g_print ("color cube: %d x %d x %d\n",
555                      image_info->nred_shades,
556                      image_info->ngreen_shades,
557                      image_info->nblue_shades);
558
559           if (!image_info->cmap_alloced)
560               image_info->cmap = gdk_colormap_get_system ();
561         }
562 #ifdef ENABLE_GRAYSCALE
563       else if (image_info->visual->type == GDK_VISUAL_GRAYSCALE)
564         {
565           image_info->cmap = gdk_colormap_new (image_info->visual, FALSE);
566           gdk_rgb_set_gray_cmap (image_info->cmap);
567           image_info->cmap_alloced = TRUE;
568         }
569 #endif
570       else
571         {
572           /* Always install colormap in direct color. */
573           if (image_info->visual->type != GDK_VISUAL_DIRECT_COLOR &&
574               image_info->visual == gdk_visual_get_system ())
575             image_info->cmap = gdk_colormap_get_system ();
576           else
577             {
578               image_info->cmap = gdk_colormap_new (image_info->visual, FALSE);
579               image_info->cmap_alloced = TRUE;
580             }
581         }
582
583       for (i = 0; i < N_IMAGES; i++)
584         static_image[i] = gdk_image_new (GDK_IMAGE_FASTEST,
585                                          image_info->visual,
586                                          IMAGE_WIDTH, IMAGE_HEIGHT);
587
588       image_info->bpp = static_image[0]->bpp;
589
590       gdk_rgb_select_conv (static_image[0]);
591
592     }
593 }
594
595 /* convert an rgb value into an X pixel code */
596 gulong
597 gdk_rgb_xpixel_from_rgb (guint32 rgb)
598 {
599   gulong pixel;
600
601   if (image_info->visual->type == GDK_VISUAL_PSEUDO_COLOR)
602     pixel = colorcube[((rgb & 0xf00000) >> 12) |
603                      ((rgb & 0xf000) >> 8) |
604                      ((rgb & 0xf0) >> 4)];
605     else
606       {
607 #ifdef VERBOSE
608         g_print ("shift, prec: r %d %d g %d %d b %d %d\n",
609                  image_info->visual->red_shift,
610                  image_info->visual->red_prec,
611                  image_info->visual->green_shift,
612                  image_info->visual->green_prec,
613                  image_info->visual->blue_shift,
614                  image_info->visual->blue_prec);
615 #endif
616
617         pixel = (((((rgb & 0xff0000) >> 16) >>
618                    (8 - image_info->visual->red_prec)) <<
619                   image_info->visual->red_shift) +
620                  ((((rgb & 0xff00) >> 8)  >>
621                    (8 - image_info->visual->green_prec)) <<
622                   image_info->visual->green_shift) +
623                  (((rgb & 0xff) >>
624                    (8 - image_info->visual->blue_prec)) <<
625                   image_info->visual->blue_shift));
626
627         ;
628       }
629
630   return pixel;
631 }
632
633 void
634 gdk_rgb_gc_set_foreground (GdkGC *gc, guint32 rgb)
635 {
636   GdkColor color;
637
638   color.pixel = gdk_rgb_xpixel_from_rgb (rgb);
639   gdk_gc_set_foreground (gc, &color);
640 }
641
642 void
643 gdk_rgb_gc_set_background (GdkGC *gc, guint32 rgb)
644 {
645   GdkColor color;
646
647   color.pixel = gdk_rgb_xpixel_from_rgb (rgb);
648   gdk_gc_set_background (gc, &color);
649 }
650
651 #ifndef WORDS_BIGENDIAN
652 #define HAIRY_CONVERT_8
653 #endif
654
655 #ifdef HAIRY_CONVERT_8
656 static void
657 gdk_rgb_convert_8 (GdkImage *image,
658                    gint x0, gint y0, gint width, gint height,
659                    guchar *buf, int rowstride,
660                    gint x_align, gint y_align, GdkRgbCmap *cmap)
661 {
662   int x, y;
663   gint bpl;
664   guchar *obuf, *obptr;
665   guchar *bptr, *bp2;
666   gint r, g, b;
667
668   bptr = buf;
669   bpl = image->bpl;
670   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
671   for (y = 0; y < height; y++)
672     {
673       bp2 = bptr;
674       obptr = obuf;
675       if (width < 8 || (((unsigned long)obuf | (unsigned long) bp2) & 3))
676         {
677           for (x = 0; x < width; x++)
678             {
679               r = *bp2++;
680               g = *bp2++;
681               b = *bp2++;
682               obptr[0] = colorcube[((r & 0xf0) << 4) |
683                                   (g & 0xf0) |
684                                   (b >> 4)];
685               obptr++;
686             }
687         }
688       else
689         {
690           for (x = 0; x < width - 3; x += 4)
691             {
692               guint32 r1b0g0r0;
693               guint32 g2r2b1g1;
694               guint32 b3g3r3b2;
695
696               r1b0g0r0 = ((guint32 *)bp2)[0];
697               g2r2b1g1 = ((guint32 *)bp2)[1];
698               b3g3r3b2 = ((guint32 *)bp2)[2];
699               ((guint32 *)obptr)[0] =
700                 colorcube[((r1b0g0r0 & 0xf0) << 4) | 
701                          ((r1b0g0r0 & 0xf000) >> 8) |
702                          ((r1b0g0r0 & 0xf00000) >> 20)] |
703                 (colorcube[((r1b0g0r0 & 0xf0000000) >> 20) |
704                           (g2r2b1g1 & 0xf0) |
705                           ((g2r2b1g1 & 0xf000) >> 12)] << 8) |
706                 (colorcube[((g2r2b1g1 & 0xf00000) >> 12) |
707                           ((g2r2b1g1 & 0xf0000000) >> 24) |
708                           ((b3g3r3b2 & 0xf0) >> 4)] << 16) |
709                 (colorcube[((b3g3r3b2 & 0xf000) >> 4) |
710                           ((b3g3r3b2 & 0xf00000) >> 16) |
711                           (b3g3r3b2 >> 28)] << 24);
712               bp2 += 12;
713               obptr += 4;
714             }
715           for (; x < width; x++)
716             {
717               r = *bp2++;
718               g = *bp2++;
719               b = *bp2++;
720               obptr[0] = colorcube[((r & 0xf0) << 4) |
721                                   (g & 0xf0) |
722                                   (b >> 4)];
723               obptr++;
724             }
725         }
726       bptr += rowstride;
727       obuf += bpl;
728     }
729 }
730 #else
731 static void
732 gdk_rgb_convert_8 (GdkImage *image,
733                    gint x0, gint y0, gint width, gint height,
734                    guchar *buf, int rowstride,
735                    gint x_align, gint y_align, GdkRgbCmap *cmap)
736 {
737   int x, y;
738   gint bpl;
739   guchar *obuf, *obptr;
740   guchar *bptr, *bp2;
741   gint r, g, b;
742
743   bptr = buf;
744   bpl = image->bpl;
745   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
746   for (y = 0; y < height; y++)
747     {
748       bp2 = bptr;
749       obptr = obuf;
750       for (x = 0; x < width; x++)
751         {
752           r = *bp2++;
753           g = *bp2++;
754           b = *bp2++;
755           obptr[0] = colorcube[((r & 0xf0) << 4) |
756                               (g & 0xf0) |
757                               (b >> 4)];
758           obptr++;
759         }
760       bptr += rowstride;
761       obuf += bpl;
762     }
763 }
764 #endif
765
766 #if 1
767
768 /* This dither table was generated by Raph Levien using patented
769    technology (US Patent 5,276,535). The dither table itself is in the
770    public domain. */
771
772 #define DM_WIDTH 128
773 #define DM_WIDTH_SHIFT 7
774 #define DM_HEIGHT 128
775 static guchar DM[128][128] =
776 {
777   { 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 },
778   { 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 },
779   { 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 },
780   { 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 },
781   { 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 },
782   { 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 },
783   { 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 },
784   { 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 },
785   { 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 },
786   { 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 },
787   { 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 },
788   { 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 },
789   { 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 },
790   { 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 },
791   { 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 },
792   { 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 },
793   { 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 },
794   { 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 },
795   { 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 },
796   { 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 },
797   { 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 },
798   { 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 },
799   { 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 },
800   { 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 },
801   { 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 },
802   { 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 },
803   { 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 },
804   { 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 },
805   { 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 },
806   { 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 },
807   { 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 },
808   { 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 },
809   { 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 },
810   { 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 },
811   { 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 },
812   { 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 },
813   { 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 },
814   { 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 },
815   { 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 },
816   { 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 },
817   { 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 },
818   { 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 },
819   { 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 },
820   { 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 },
821   { 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 },
822   { 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 },
823   { 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 },
824   { 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 },
825   { 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 },
826   { 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 },
827   { 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 },
828   { 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 },
829   { 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 },
830   { 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 },
831   { 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 },
832   { 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 },
833   { 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 },
834   { 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 },
835   { 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 },
836   { 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 },
837   { 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 },
838   { 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 },
839   { 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 },
840   { 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 },
841   { 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 },
842   { 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 },
843   { 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 },
844   { 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 },
845   { 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 },
846   { 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 },
847   { 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 },
848   { 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 },
849   { 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 },
850   { 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 },
851   { 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 },
852   { 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 },
853   { 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 },
854   { 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 },
855   { 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 },
856   { 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 },
857   { 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 },
858   { 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 },
859   { 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 },
860   { 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 },
861   { 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 },
862   { 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 },
863   { 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 },
864   { 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 },
865   { 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 },
866   { 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 },
867   { 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 },
868   { 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 },
869   { 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 },
870   { 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 },
871   { 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 },
872   { 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 },
873   { 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 },
874   { 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 },
875   { 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 },
876   { 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 },
877   { 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 },
878   { 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 },
879   { 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 },
880   { 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, 64, 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 },
881   { 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 },
882   { 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 },
883   { 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 },
884   { 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 },
885   { 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 },
886   { 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 },
887   { 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 },
888   { 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 },
889   { 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 },
890   { 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 },
891   { 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 },
892   { 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 },
893   { 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 },
894   { 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 },
895   { 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 },
896   { 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 },
897   { 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 },
898   { 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 },
899   { 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 },
900   { 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 },
901   { 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 },
902   { 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 },
903   { 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 },
904   { 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 },
905 };
906
907 #else
908 #define DM_WIDTH 8
909 #define DM_WIDTH_SHIFT 3
910 #define DM_HEIGHT 8
911 static guchar DM[8][8] =
912 {
913   { 0,  32, 8,  40, 2,  34, 10, 42 },
914   { 48, 16, 56, 24, 50, 18, 58, 26 },
915   { 12, 44, 4,  36, 14, 46, 6,  38 },
916   { 60, 28, 52, 20, 62, 30, 54, 22 },
917   { 3,  35, 11, 43, 1,  33, 9,  41 },
918   { 51, 19, 59, 27, 49, 17, 57, 25 },
919   { 15, 47, 7,  39, 13, 45, 5,  37 },
920   { 63, 31, 55, 23, 61, 29, 53, 21 }
921 };
922 #endif
923
924 static guint32 *DM_565 = NULL;
925
926 static void
927 gdk_rgb_preprocess_dm_565 (void)
928 {
929   int i;
930   guint32 dith;
931
932   if (DM_565 == NULL)
933     {
934       DM_565 = g_new (guint32, DM_WIDTH * DM_HEIGHT);
935       for (i = 0; i < DM_WIDTH * DM_HEIGHT; i++)
936         {
937           dith = DM[0][i] >> 3;
938           DM_565[i] = (dith << 20) | dith | (((7 - dith) >> 1) << 10);
939 #ifdef VERBOSE
940           g_print ("%i %x %x\n", i, dith, DM_565[i]);
941 #endif
942         }
943     }
944 }
945
946 static void
947 gdk_rgb_convert_8_d666 (GdkImage *image,
948                         gint x0, gint y0, gint width, gint height,
949                         guchar *buf, int rowstride,
950                         gint x_align, gint y_align, GdkRgbCmap *cmap)
951 {
952   int x, y;
953   gint bpl;
954   guchar *obuf, *obptr;
955   guchar *bptr, *bp2;
956   gint r, g, b;
957   guchar *dmp;
958   gint dith;
959
960   bptr = buf;
961   bpl = image->bpl;
962   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
963   for (y = 0; y < height; y++)
964     {
965       dmp = DM[(y_align + y) & (DM_HEIGHT - 1)];
966       bp2 = bptr;
967       obptr = obuf;
968       for (x = 0; x < width; x++)
969         {
970           r = *bp2++;
971           g = *bp2++;
972           b = *bp2++;
973           dith = (dmp[(x_align + x) & (DM_WIDTH - 1)] << 2) | 7;
974           r = ((r * 5) + dith) >> 8;
975           g = ((g * 5) + (262 - dith)) >> 8;
976           b = ((b * 5) + dith) >> 8;
977           obptr[0] = colorcube_d[(r << 6) | (g << 3) | b];
978           obptr++;
979         }
980       bptr += rowstride;
981       obuf += bpl;
982     }
983 }
984
985 static void
986 gdk_rgb_convert_8_d (GdkImage *image,
987                      gint x0, gint y0, gint width, gint height,
988                      guchar *buf, int rowstride,
989                      gint x_align, gint y_align,
990                      GdkRgbCmap *cmap)
991 {
992   int x, y;
993   gint bpl;
994   guchar *obuf, *obptr;
995   guchar *bptr, *bp2;
996   gint r, g, b;
997   guchar *dmp;
998   gint dith;
999   gint rs, gs, bs;
1000
1001   bptr = buf;
1002   bpl = image->bpl;
1003   rs = image_info->nred_shades - 1;
1004   gs = image_info->ngreen_shades - 1;
1005   bs = image_info->nblue_shades - 1;
1006   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
1007   for (y = 0; y < height; y++)
1008     {
1009       dmp = DM[(y_align + y) & (DM_HEIGHT - 1)];
1010       bp2 = bptr;
1011       obptr = obuf;
1012       for (x = 0; x < width; x++)
1013         {
1014           r = *bp2++;
1015           g = *bp2++;
1016           b = *bp2++;
1017           dith = (dmp[(x_align + x) & (DM_WIDTH - 1)] << 2) | 7;
1018           r = ((r * rs) + dith) >> 8;
1019           g = ((g * gs) + (262 - dith)) >> 8;
1020           b = ((b * bs) + dith) >> 8;
1021           obptr[0] = colorcube_d[(r << 6) | (g << 3) | b];
1022           obptr++;
1023         }
1024       bptr += rowstride;
1025       obuf += bpl;
1026     }
1027 }
1028
1029 static void
1030 gdk_rgb_convert_8_indexed (GdkImage *image,
1031                            gint x0, gint y0, gint width, gint height,
1032                            guchar *buf, int rowstride,
1033                            gint x_align, gint y_align, GdkRgbCmap *cmap)
1034 {
1035   int x, y;
1036   gint bpl;
1037   guchar *obuf, *obptr;
1038   guchar *bptr, *bp2;
1039   guchar c;
1040   guchar *lut;
1041
1042   lut = cmap->lut;
1043   bptr = buf;
1044   bpl = image->bpl;
1045   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
1046   for (y = 0; y < height; y++)
1047     {
1048       bp2 = bptr;
1049       obptr = obuf;
1050       for (x = 0; x < width; x++)
1051         {
1052           c = *bp2++;
1053           obptr[0] = lut[c];
1054           obptr++;
1055         }
1056       bptr += rowstride;
1057       obuf += bpl;
1058     }
1059 }
1060
1061 static void
1062 gdk_rgb_convert_gray8 (GdkImage *image,
1063                        gint x0, gint y0, gint width, gint height,
1064                        guchar *buf, int rowstride,
1065                        gint x_align, gint y_align, GdkRgbCmap *cmap)
1066 {
1067   int x, y;
1068   gint bpl;
1069   guchar *obuf, *obptr;
1070   guchar *bptr, *bp2;
1071   gint r, g, b;
1072
1073   bptr = buf;
1074   bpl = image->bpl;
1075   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
1076   for (y = 0; y < height; y++)
1077     {
1078       bp2 = bptr;
1079       obptr = obuf;
1080       for (x = 0; x < width; x++)
1081         {
1082           r = *bp2++;
1083           g = *bp2++;
1084           b = *bp2++;
1085           obptr[0] = (g + ((b + r) >> 1)) >> 1;
1086           obptr++;
1087         }
1088       bptr += rowstride;
1089       obuf += bpl;
1090     }
1091 }
1092
1093 static void
1094 gdk_rgb_convert_gray8_gray (GdkImage *image,
1095                             gint x0, gint y0, gint width, gint height,
1096                             guchar *buf, int rowstride,
1097                             gint x_align, gint y_align, GdkRgbCmap *cmap)
1098 {
1099   int y;
1100   gint bpl;
1101   guchar *obuf;
1102   guchar *bptr;
1103
1104   bptr = buf;
1105   bpl = image->bpl;
1106   obuf = ((guchar *)image->mem) + y0 * bpl + x0;
1107   for (y = 0; y < height; y++)
1108     {
1109       memcpy (obuf, bptr, width);
1110       bptr += rowstride;
1111       obuf += bpl;
1112     }
1113 }
1114
1115 #ifndef WORDS_BIGENDIAN
1116 #define HAIRY_CONVERT_565
1117 #endif
1118
1119 #ifdef HAIRY_CONVERT_565
1120 /* Render a 24-bit RGB image in buf into the GdkImage, without dithering.
1121    This assumes native byte ordering - what should really be done is to
1122    check whether static_image->byte_order is consistent with the _ENDIAN
1123    config flag, and if not, use a different function.
1124
1125    This one is even faster than the one below - its inner loop loads 3
1126    words (i.e. 4 24-bit pixels), does a lot of shifting and masking,
1127    then writes 2 words. */
1128 static void
1129 gdk_rgb_convert_565 (GdkImage *image,
1130                      gint x0, gint y0, gint width, gint height,
1131                      guchar *buf, int rowstride,
1132                      gint x_align, gint y_align, GdkRgbCmap *cmap)
1133 {
1134   int x, y;
1135   guchar *obuf, *obptr;
1136   gint bpl;
1137   guchar *bptr, *bp2;
1138   guchar r, g, b;
1139
1140   bptr = buf;
1141   bpl = image->bpl;
1142   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1143   for (y = 0; y < height; y++)
1144     {
1145       bp2 = bptr;
1146       obptr = obuf;
1147       if (width < 8 || (((unsigned long)obuf | (unsigned long) bp2) & 3))
1148         {
1149           for (x = 0; x < width; x++)
1150             {
1151               r = *bp2++;
1152               g = *bp2++;
1153               b = *bp2++;
1154               ((guint16 *)obptr)[0] = ((r & 0xf8) << 8) |
1155                 ((g & 0xfc) << 3) |
1156                 (b >> 3);
1157               obptr += 2;
1158             }
1159         }
1160       else
1161         {
1162           for (x = 0; x < width - 3; x += 4)
1163             {
1164               guint32 r1b0g0r0;
1165               guint32 g2r2b1g1;
1166               guint32 b3g3r3b2;
1167
1168               r1b0g0r0 = ((guint32 *)bp2)[0];
1169               g2r2b1g1 = ((guint32 *)bp2)[1];
1170               b3g3r3b2 = ((guint32 *)bp2)[2];
1171               ((guint32 *)obptr)[0] =
1172                 ((r1b0g0r0 & 0xf8) << 8) |
1173                 ((r1b0g0r0 & 0xfc00) >> 5) |
1174                 ((r1b0g0r0 & 0xf80000) >> 19) |
1175                 (r1b0g0r0 & 0xf8000000) |
1176                 ((g2r2b1g1 & 0xfc) << 19) |
1177                 ((g2r2b1g1 & 0xf800) << 5);
1178               ((guint32 *)obptr)[1] =
1179                 ((g2r2b1g1 & 0xf80000) >> 8) |
1180                 ((g2r2b1g1 & 0xfc000000) >> 21) |
1181                 ((b3g3r3b2 & 0xf8) >> 3) |
1182                 ((b3g3r3b2 & 0xf800) << 16) |
1183                 ((b3g3r3b2 & 0xfc0000) << 3) |
1184                 ((b3g3r3b2 & 0xf8000000) >> 11);
1185               bp2 += 12;
1186               obptr += 8;
1187             }
1188           for (; x < width; x++)
1189             {
1190               r = *bp2++;
1191               g = *bp2++;
1192               b = *bp2++;
1193               ((guint16 *)obptr)[0] = ((r & 0xf8) << 8) |
1194                 ((g & 0xfc) << 3) |
1195                 (b >> 3);
1196               obptr += 2;
1197             }
1198         }
1199       bptr += rowstride;
1200       obuf += bpl;
1201     }
1202 }
1203 #else
1204 /* Render a 24-bit RGB image in buf into the GdkImage, without dithering.
1205    This assumes native byte ordering - what should really be done is to
1206    check whether static_image->byte_order is consistent with the _ENDIAN
1207    config flag, and if not, use a different function.
1208
1209    This routine is faster than the one included with Gtk 1.0 for a number
1210    of reasons:
1211
1212    1. Shifting instead of lookup tables (less memory traffic).
1213
1214    2. Much less register pressure, especially because shifts are
1215    in the code.
1216
1217    3. A memcpy is avoided (i.e. the transfer function).
1218
1219    4. On big-endian architectures, byte swapping is avoided.
1220
1221    That said, it wouldn't be hard to make it even faster - just make an
1222    inner loop that reads 3 words (i.e. 4 24-bit pixels), does a lot of
1223    shifting and masking, then writes 2 words.
1224 */
1225 static void
1226 gdk_rgb_convert_565 (GdkImage *image,
1227                      gint x0, gint y0, gint width, gint height,
1228                      guchar *buf, int rowstride,
1229                      gint x_align, gint y_align, GdkRgbCmap *cmap)
1230 {
1231   int x, y;
1232   guchar *obuf;
1233   gint bpl;
1234   guchar *bptr, *bp2;
1235   guchar r, g, b;
1236
1237   bptr = buf;
1238   bpl = image->bpl;
1239   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1240   for (y = 0; y < height; y++)
1241     {
1242       bp2 = bptr;
1243       for (x = 0; x < width; x++)
1244         {
1245           r = *bp2++;
1246           g = *bp2++;
1247           b = *bp2++;
1248           ((unsigned short *)obuf)[x] = ((r & 0xf8) << 8) |
1249             ((g & 0xfc) << 3) |
1250             (b >> 3);
1251         }
1252       bptr += rowstride;
1253       obuf += bpl;
1254     }
1255 }
1256 #endif
1257
1258 #ifdef HAIRY_CONVERT_565
1259 static void
1260 gdk_rgb_convert_565_gray (GdkImage *image,
1261                           gint x0, gint y0, gint width, gint height,
1262                           guchar *buf, int rowstride,
1263                           gint x_align, gint y_align, GdkRgbCmap *cmap)
1264 {
1265   int x, y;
1266   guchar *obuf, *obptr;
1267   gint bpl;
1268   guchar *bptr, *bp2;
1269   guchar g;
1270
1271   bptr = buf;
1272   bpl = image->bpl;
1273   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1274   for (y = 0; y < height; y++)
1275     {
1276       bp2 = bptr;
1277       obptr = obuf;
1278       if (width < 8 || (((unsigned long)obuf | (unsigned long) bp2) & 3))
1279         {
1280           for (x = 0; x < width; x++)
1281             {
1282               g = *bp2++;
1283               ((guint16 *)obptr)[0] = ((g & 0xf8) << 8) |
1284                 ((g & 0xfc) << 3) |
1285                 (g >> 3);
1286               obptr += 2;
1287             }
1288         }
1289       else
1290         {
1291           for (x = 0; x < width - 3; x += 4)
1292             {
1293               guint32 g3g2g1g0;
1294
1295               g3g2g1g0 = ((guint32 *)bp2)[0];
1296               ((guint32 *)obptr)[0] =
1297                 ((g3g2g1g0 & 0xf8) << 8) |
1298                 ((g3g2g1g0 & 0xfc) << 3) |
1299                 ((g3g2g1g0 & 0xf8) >> 3) |
1300                 (g3g2g1g0 & 0xf800) << 16 |
1301                 ((g3g2g1g0 & 0xfc00) << 11) |
1302                 ((g3g2g1g0 & 0xf800) << 5);
1303               ((guint32 *)obptr)[1] =
1304                 ((g3g2g1g0 & 0xf80000) >> 8) |
1305                 ((g3g2g1g0 & 0xfc0000) >> 13) |
1306                 ((g3g2g1g0 & 0xf80000) >> 19) |
1307                 (g3g2g1g0 & 0xf8000000) |
1308                 ((g3g2g1g0 & 0xfc000000) >> 5) |
1309                 ((g3g2g1g0 & 0xf8000000) >> 11);
1310               bp2 += 4;
1311               obptr += 8;
1312             }
1313           for (; x < width; x++)
1314             {
1315               g = *bp2++;
1316               ((guint16 *)obptr)[0] = ((g & 0xf8) << 8) |
1317                 ((g & 0xfc) << 3) |
1318                 (g >> 3);
1319               obptr += 2;
1320             }
1321         }
1322       bptr += rowstride;
1323       obuf += bpl;
1324     }
1325 }
1326 #else
1327 static void
1328 gdk_rgb_convert_565_gray (GdkImage *image,
1329                           gint x0, gint y0, gint width, gint height,
1330                           guchar *buf, int rowstride,
1331                           gint x_align, gint y_align, GdkRgbCmap *cmap)
1332 {
1333   int x, y;
1334   guchar *obuf;
1335   gint bpl;
1336   guchar *bptr, *bp2;
1337   guchar g;
1338
1339   bptr = buf;
1340   bpl = image->bpl;
1341   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1342   for (y = 0; y < height; y++)
1343     {
1344       bp2 = bptr;
1345       for (x = 0; x < width; x++)
1346         {
1347           g = *bp2++;
1348           ((guint16 *)obuf)[x] = ((g & 0xf8) << 8) |
1349             ((g & 0xfc) << 3) |
1350             (g >> 3);
1351         }
1352       bptr += rowstride;
1353       obuf += bpl;
1354     }
1355 }
1356 #endif
1357
1358 static void
1359 gdk_rgb_convert_565_br (GdkImage *image,
1360                         gint x0, gint y0, gint width, gint height,
1361                         guchar *buf, int rowstride,
1362                         gint x_align, gint y_align, GdkRgbCmap *cmap)
1363 {
1364   int x, y;
1365   guchar *obuf;
1366   gint bpl;
1367   guchar *bptr, *bp2;
1368   guchar r, g, b;
1369
1370   bptr = buf;
1371   bpl = image->bpl;
1372   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1373   for (y = 0; y < height; y++)
1374     {
1375       bp2 = bptr;
1376       for (x = 0; x < width; x++)
1377         {
1378           r = *bp2++;
1379           g = *bp2++;
1380           b = *bp2++;
1381           /* final word is:
1382              g4 g3 g2 b7 b6 b5 b4 b3  r7 r6 r5 r4 r3 g7 g6 g5
1383            */
1384           ((unsigned short *)obuf)[x] = (r & 0xf8) |
1385             ((g & 0xe0) >> 5) |
1386             ((g & 0x1c) << 11) |
1387             ((b & 0xf8) << 5);
1388         }
1389       bptr += rowstride;
1390       obuf += bpl;
1391     }
1392 }
1393
1394 /* Thanks to Ray Lehtiniemi for a patch that resulted in a ~25% speedup
1395    in this mode. */
1396 #ifdef HAIRY_CONVERT_565
1397 static void
1398 gdk_rgb_convert_565_d (GdkImage *image,
1399                      gint x0, gint y0, gint width, gint height,
1400                      guchar *buf, int rowstride,
1401                      gint x_align, gint y_align, GdkRgbCmap *cmap)
1402 {
1403   /* Now this is what I'd call some highly tuned code! */
1404   int x, y;
1405   guchar *obuf, *obptr;
1406   gint bpl;
1407   guchar *bptr, *bp2;
1408
1409   width += x_align;
1410   height += y_align;
1411   
1412   bptr = buf;
1413   bpl = image->bpl;
1414   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1415   for (y = y_align; y < height; y++)
1416     {
1417       guint32 *dmp = DM_565 + ((y & (DM_HEIGHT - 1)) << DM_WIDTH_SHIFT);
1418       bp2 = bptr;
1419       obptr = obuf;
1420       if (width < 8 || (((unsigned long)obuf | (unsigned long) bp2) & 3))
1421         {
1422           for (x = x_align; x < width; x++)
1423             {
1424               gint32 rgb = *bp2++ << 20;
1425               rgb += *bp2++ << 10;
1426               rgb += *bp2++;
1427               rgb += dmp[x & (DM_WIDTH - 1)];
1428               rgb += 0x10040100
1429                 - ((rgb & 0x1e0001e0) >> 5)
1430                 - ((rgb & 0x00070000) >> 6);
1431
1432               ((unsigned short *)obuf)[0] =
1433                 ((rgb & 0x0f800000) >> 12) |
1434                 ((rgb & 0x0003f000) >> 7) |
1435                 ((rgb & 0x000000f8) >> 3);
1436               obptr += 2;
1437             }
1438         }
1439       else
1440         {
1441           for (x = x_align; x < width - 3; x += 4)
1442             {
1443               guint32 r1b0g0r0;
1444               guint32 g2r2b1g1;
1445               guint32 b3g3r3b2;
1446               guint32 rgb02, rgb13;
1447
1448               r1b0g0r0 = ((guint32 *)bp2)[0];
1449               g2r2b1g1 = ((guint32 *)bp2)[1];
1450               b3g3r3b2 = ((guint32 *)bp2)[2];
1451               rgb02 =
1452                 ((r1b0g0r0 & 0xff) << 20) +
1453                 ((r1b0g0r0 & 0xff00) << 2) +
1454                 ((r1b0g0r0 & 0xff0000) >> 16) +
1455                 dmp[x & (DM_WIDTH - 1)];
1456               rgb02 += 0x10040100
1457                 - ((rgb02 & 0x1e0001e0) >> 5)
1458                 - ((rgb02 & 0x00070000) >> 6);
1459               rgb13 =
1460                 ((r1b0g0r0 & 0xff000000) >> 4) +
1461                 ((g2r2b1g1 & 0xff) << 10) +
1462                 ((g2r2b1g1 & 0xff00) >> 8) +
1463                 dmp[(x + 1) & (DM_WIDTH - 1)];
1464               rgb13 += 0x10040100
1465                 - ((rgb13 & 0x1e0001e0) >> 5)
1466                 - ((rgb13 & 0x00070000) >> 6);
1467               ((guint32 *)obptr)[0] =
1468                 ((rgb02 & 0x0f800000) >> 12) |
1469                 ((rgb02 & 0x0003f000) >> 7) |
1470                 ((rgb02 & 0x000000f8) >> 3) |
1471                 ((rgb13 & 0x0f800000) << 4) |
1472                 ((rgb13 & 0x0003f000) << 9) |
1473                 ((rgb13 & 0x000000f8) << 13);
1474               rgb02 =
1475                 ((g2r2b1g1 & 0xff0000) << 4) +
1476                 ((g2r2b1g1 & 0xff000000) >> 14) +
1477                 (b3g3r3b2 & 0xff) +
1478                 dmp[(x + 2) & (DM_WIDTH - 1)];
1479               rgb02 += 0x10040100
1480                 - ((rgb02 & 0x1e0001e0) >> 5)
1481                 - ((rgb02 & 0x00070000) >> 6);
1482               rgb13 =
1483                 ((b3g3r3b2 & 0xff00) << 12) +
1484                 ((b3g3r3b2 & 0xff0000) >> 6) +
1485                 ((b3g3r3b2 & 0xff000000) >> 24) +
1486                 dmp[(x + 3) & (DM_WIDTH - 1)];
1487               rgb13 += 0x10040100
1488                 - ((rgb13 & 0x1e0001e0) >> 5)
1489                 - ((rgb13 & 0x00070000) >> 6);
1490               ((guint32 *)obptr)[1] =
1491                 ((rgb02 & 0x0f800000) >> 12) |
1492                 ((rgb02 & 0x0003f000) >> 7) |
1493                 ((rgb02 & 0x000000f8) >> 3) |
1494                 ((rgb13 & 0x0f800000) << 4) |
1495                 ((rgb13 & 0x0003f000) << 9) |
1496                 ((rgb13 & 0x000000f8) << 13);
1497               bp2 += 12;
1498               obptr += 8;
1499             }
1500           for (; x < width; x++)
1501             {
1502               gint32 rgb = *bp2++ << 20;
1503               rgb += *bp2++ << 10;
1504               rgb += *bp2++;
1505               rgb += dmp[x & (DM_WIDTH - 1)];
1506               rgb += 0x10040100
1507                 - ((rgb & 0x1e0001e0) >> 5)
1508                 - ((rgb & 0x00070000) >> 6);
1509
1510               ((unsigned short *)obuf)[0] =
1511                 ((rgb & 0x0f800000) >> 12) |
1512                 ((rgb & 0x0003f000) >> 7) |
1513                 ((rgb & 0x000000f8) >> 3);
1514               obptr += 2;
1515             }
1516         }
1517       bptr += rowstride;
1518       obuf += bpl;
1519     }
1520 }
1521 #else
1522 static void
1523 gdk_rgb_convert_565_d (GdkImage *image,
1524                        gint x0, gint y0, gint width, gint height,
1525                        guchar *buf, int rowstride,
1526                        gint x_align, gint y_align, GdkRgbCmap *cmap)
1527 {
1528   int x, y;
1529   guchar *obuf;
1530   gint bpl;
1531   guchar *bptr;
1532
1533   width += x_align;
1534   height += y_align;
1535   
1536   bptr = buf;
1537   bpl = image->bpl;
1538   obuf = ((guchar *)image->mem) + y0 * bpl + (x0 - x_align) * 2;
1539
1540   for (y = y_align; y < height; y++)
1541     {
1542       guint32 *dmp = DM_565 + ((y & (DM_HEIGHT - 1)) << DM_WIDTH_SHIFT);
1543       guchar *bp2 = bptr;
1544
1545       for (x = x_align; x < width; x++)
1546         {
1547           gint32 rgb = *bp2++ << 20;
1548           rgb += *bp2++ << 10;
1549           rgb += *bp2++;
1550           rgb += dmp[x & (DM_WIDTH - 1)];
1551           rgb += 0x10040100
1552             - ((rgb & 0x1e0001e0) >> 5)
1553             - ((rgb & 0x00070000) >> 6);
1554
1555           ((unsigned short *)obuf)[x] =
1556             ((rgb & 0x0f800000) >> 12) |
1557             ((rgb & 0x0003f000) >> 7) |
1558             ((rgb & 0x000000f8) >> 3);
1559         }
1560
1561       bptr += rowstride;
1562       obuf += bpl;
1563     }
1564 }
1565 #endif
1566
1567 static void
1568 gdk_rgb_convert_555 (GdkImage *image,
1569                      gint x0, gint y0, gint width, gint height,
1570                      guchar *buf, int rowstride,
1571                      gint x_align, gint y_align, GdkRgbCmap *cmap)
1572 {
1573   int x, y;
1574   guchar *obuf;
1575   gint bpl;
1576   guchar *bptr, *bp2;
1577   guchar r, g, b;
1578
1579   bptr = buf;
1580   bpl = image->bpl;
1581   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1582   for (y = 0; y < height; y++)
1583     {
1584       bp2 = bptr;
1585       for (x = 0; x < width; x++)
1586         {
1587           r = *bp2++;
1588           g = *bp2++;
1589           b = *bp2++;
1590           ((unsigned short *)obuf)[x] = ((r & 0xf8) << 7) |
1591             ((g & 0xf8) << 2) |
1592             (b >> 3);
1593         }
1594       bptr += rowstride;
1595       obuf += bpl;
1596     }
1597 }
1598
1599 static void
1600 gdk_rgb_convert_555_br (GdkImage *image,
1601                         gint x0, gint y0, gint width, gint height,
1602                         guchar *buf, int rowstride,
1603                         gint x_align, gint y_align, GdkRgbCmap *cmap)
1604 {
1605   int x, y;
1606   guchar *obuf;
1607   gint bpl;
1608   guchar *bptr, *bp2;
1609   guchar r, g, b;
1610
1611   bptr = buf;
1612   bpl = image->bpl;
1613   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 2;
1614   for (y = 0; y < height; y++)
1615     {
1616       bp2 = bptr;
1617       for (x = 0; x < width; x++)
1618         {
1619           r = *bp2++;
1620           g = *bp2++;
1621           b = *bp2++;
1622           /* final word is:
1623              g5 g4 g3 b7 b6 b5 b4 b3  0 r7 r6 r5 r4 r3 g7 g6
1624            */
1625           ((unsigned short *)obuf)[x] = ((r & 0xf8) >> 1) |
1626             ((g & 0xc0) >> 6) |
1627             ((g & 0x18) << 10) |
1628             ((b & 0xf8) << 5);
1629         }
1630       bptr += rowstride;
1631       obuf += bpl;
1632     }
1633 }
1634
1635 static void
1636 gdk_rgb_convert_888_msb (GdkImage *image,
1637                          gint x0, gint y0, gint width, gint height,
1638                          guchar *buf, int rowstride,
1639                          gint x_align, gint y_align, GdkRgbCmap *cmap)
1640 {
1641   int y;
1642   guchar *obuf;
1643   gint bpl;
1644   guchar *bptr;
1645
1646   bptr = buf;
1647   bpl = image->bpl;
1648   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 3;
1649   for (y = 0; y < height; y++)
1650     {
1651       memcpy (obuf, bptr, width + width + width);
1652       bptr += rowstride;
1653       obuf += bpl;
1654     }
1655 }
1656
1657 /* todo: optimize this */
1658 #ifndef WORDS_BIGENDIAN
1659 #define HAIRY_CONVERT_888
1660 #endif
1661
1662 #ifdef HAIRY_CONVERT_888
1663 static void
1664 gdk_rgb_convert_888_lsb (GdkImage *image,
1665                          gint x0, gint y0, gint width, gint height,
1666                          guchar *buf, int rowstride,
1667                          gint x_align, gint y_align, GdkRgbCmap *cmap)
1668 {
1669   int x, y;
1670   guchar *obuf, *obptr;
1671   gint bpl;
1672   guchar *bptr, *bp2;
1673   int r, g, b;
1674
1675   bptr = buf;
1676   bpl = image->bpl;
1677   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 3;
1678   for (y = 0; y < height; y++)
1679     {
1680       bp2 = bptr;
1681       obptr = obuf;
1682       if (width < 8 || (((unsigned long)obuf | (unsigned long) bp2) & 3))
1683         {
1684           for (x = 0; x < width; x++)
1685             {
1686               r = bp2[0];
1687               g = bp2[1];
1688               b = bp2[2];
1689               *obptr++ = b;
1690               *obptr++ = g;
1691               *obptr++ = r;
1692               bp2 += 3;
1693             }
1694         }
1695       else
1696         {
1697           for (x = 0; x < width - 3; x += 4)
1698             {
1699               guint32 r1b0g0r0;
1700               guint32 g2r2b1g1;
1701               guint32 b3g3r3b2;
1702
1703               r1b0g0r0 = ((guint32 *)bp2)[0];
1704               g2r2b1g1 = ((guint32 *)bp2)[1];
1705               b3g3r3b2 = ((guint32 *)bp2)[2];
1706               ((guint32 *)obptr)[0] =
1707                 (r1b0g0r0 & 0xff00) |
1708                 ((r1b0g0r0 & 0xff0000) >> 16) |
1709                 (((g2r2b1g1 & 0xff00) | (r1b0g0r0 & 0xff)) << 16);
1710               ((guint32 *)obptr)[1] =
1711                 (g2r2b1g1 & 0xff0000ff) |
1712                 ((r1b0g0r0 & 0xff000000) >> 16) |
1713                 ((b3g3r3b2 & 0xff) << 16);
1714               ((guint32 *)obptr)[2] =
1715                 (((g2r2b1g1 & 0xff0000) | (b3g3r3b2 & 0xff000000)) >> 16) |
1716                 ((b3g3r3b2 & 0xff00) << 16) |
1717                 ((b3g3r3b2 & 0xff0000));
1718               bp2 += 12;
1719               obptr += 12;
1720             }
1721           for (; x < width; x++)
1722             {
1723               r = bp2[0];
1724               g = bp2[1];
1725               b = bp2[2];
1726               *obptr++ = b;
1727               *obptr++ = g;
1728               *obptr++ = r;
1729               bp2 += 3;
1730             }
1731         }
1732       bptr += rowstride;
1733       obuf += bpl;
1734     }
1735 }
1736 #else
1737 static void
1738 gdk_rgb_convert_888_lsb (GdkImage *image,
1739                          gint x0, gint y0, gint width, gint height,
1740                          guchar *buf, int rowstride,
1741                          gint x_align, gint y_align, GdkRgbCmap *cmap)
1742 {
1743   int x, y;
1744   guchar *obuf;
1745   gint bpl;
1746   guchar *bptr, *bp2;
1747   int r, g, b;
1748
1749   bptr = buf;
1750   bpl = image->bpl;
1751   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 3;
1752   for (y = 0; y < height; y++)
1753     {
1754       bp2 = bptr;
1755       for (x = 0; x < width; x++)
1756         {
1757           r = bp2[0];
1758           g = bp2[1];
1759           b = bp2[2];
1760           obuf[x * 3] = b;
1761           obuf[x * 3 + 1] = g;
1762           obuf[x * 3 + 2] = r;
1763           bp2 += 3;
1764         }
1765       bptr += rowstride;
1766       obuf += bpl;
1767     }
1768 }
1769 #endif
1770
1771 /* convert 24-bit packed to 32-bit unpacked */
1772 /* todo: optimize this */
1773 static void
1774 gdk_rgb_convert_0888 (GdkImage *image,
1775                       gint x0, gint y0, gint width, gint height,
1776                       guchar *buf, int rowstride,
1777                       gint x_align, gint y_align, GdkRgbCmap *cmap)
1778 {
1779   int x, y;
1780   guchar *obuf;
1781   gint bpl;
1782   guchar *bptr, *bp2;
1783   int r, g, b;
1784
1785   bptr = buf;
1786   bpl = image->bpl;
1787   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 4;
1788   for (y = 0; y < height; y++)
1789     {
1790       bp2 = bptr;
1791       for (x = 0; x < width; x++)
1792         {
1793           r = bp2[0];
1794           g = bp2[1];
1795           b = bp2[2];
1796           ((unsigned long *)obuf)[x] = (r << 16) | (g << 8) | b;
1797           bp2 += 3;
1798         }
1799       bptr += rowstride;
1800       obuf += bpl;
1801     }
1802 }
1803
1804 static void
1805 gdk_rgb_convert_0888_br (GdkImage *image,
1806                          gint x0, gint y0, gint width, gint height,
1807                          guchar *buf, int rowstride,
1808                          gint x_align, gint y_align, GdkRgbCmap *cmap)
1809 {
1810   int x, y;
1811   guchar *obuf;
1812   gint bpl;
1813   guchar *bptr, *bp2;
1814   int r, g, b;
1815
1816   bptr = buf;
1817   bpl = image->bpl;
1818   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * 4;
1819   for (y = 0; y < height; y++)
1820     {
1821       bp2 = bptr;
1822       for (x = 0; x < width; x++)
1823         {
1824           r = bp2[0];
1825           g = bp2[1];
1826           b = bp2[2];
1827           ((unsigned long *)obuf)[x] = (b << 24) | (g << 16) | (r << 8);
1828           bp2 += 3;
1829         }
1830       bptr += rowstride;
1831       obuf += bpl;
1832     }
1833 }
1834
1835 /* Generic truecolor/directcolor conversion function. Slow, but these
1836    are oddball modes. */
1837 static void
1838 gdk_rgb_convert_truecolor_lsb (GdkImage *image,
1839                                gint x0, gint y0, gint width, gint height,
1840                                guchar *buf, int rowstride,
1841                                gint x_align, gint y_align,
1842                                GdkRgbCmap *cmap)
1843 {
1844   int x, y;
1845   guchar *obuf, *obptr;
1846   gint bpl;
1847   guchar *bptr, *bp2;
1848   gint r, g, b;
1849   gint r_right, r_left;
1850   gint g_right, g_left;
1851   gint b_right, b_left;
1852   gint bpp;
1853   guint32 pixel;
1854   gint i;
1855
1856   r_right = 8 - image_info->visual->red_prec;
1857   r_left = image_info->visual->red_shift;
1858   g_right = 8 - image_info->visual->green_prec;
1859   g_left = image_info->visual->green_shift;
1860   b_right = 8 - image_info->visual->blue_prec;
1861   b_left = image_info->visual->blue_shift;
1862   bpp = image_info->bpp;
1863   bptr = buf;
1864   bpl = image->bpl;
1865   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * bpp;
1866   for (y = 0; y < height; y++)
1867     {
1868       obptr = obuf;
1869       bp2 = bptr;
1870       for (x = 0; x < width; x++)
1871         {
1872           r = bp2[0];
1873           g = bp2[1];
1874           b = bp2[2];
1875           pixel = ((r >> r_right) << r_left) |
1876             ((g >> g_right) << g_left) |
1877             ((b >> b_right) << b_left);
1878           for (i = 0; i < bpp; i++)
1879             {
1880               *obptr++ = pixel & 0xff;
1881               pixel >>= 8;
1882             }
1883           bp2 += 3;
1884         }
1885       bptr += rowstride;
1886       obuf += bpl;
1887     }
1888 }
1889
1890 static void
1891 gdk_rgb_convert_truecolor_lsb_d (GdkImage *image,
1892                                  gint x0, gint y0, gint width, gint height,
1893                                  guchar *buf, int rowstride,
1894                                  gint x_align, gint y_align,
1895                                  GdkRgbCmap *cmap)
1896 {
1897   int x, y;
1898   guchar *obuf, *obptr;
1899   gint bpl;
1900   guchar *bptr, *bp2;
1901   gint r, g, b;
1902   gint r_right, r_left, r_prec;
1903   gint g_right, g_left, g_prec;
1904   gint b_right, b_left, b_prec;
1905   gint bpp;
1906   guint32 pixel;
1907   gint i;
1908   gint dith;
1909   gint r1, g1, b1;
1910   guchar *dmp;
1911
1912   r_right = 8 - image_info->visual->red_prec;
1913   r_left = image_info->visual->red_shift;
1914   r_prec = image_info->visual->red_prec;
1915   g_right = 8 - image_info->visual->green_prec;
1916   g_left = image_info->visual->green_shift;
1917   g_prec = image_info->visual->green_prec;
1918   b_right = 8 - image_info->visual->blue_prec;
1919   b_left = image_info->visual->blue_shift;
1920   b_prec = image_info->visual->blue_prec;
1921   bpp = image_info->bpp;
1922   bptr = buf;
1923   bpl = image->bpl;
1924   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * bpp;
1925   for (y = 0; y < height; y++)
1926     {
1927       dmp = DM[(y_align + y) & (DM_HEIGHT - 1)];
1928       obptr = obuf;
1929       bp2 = bptr;
1930       for (x = 0; x < width; x++)
1931         {
1932           r = bp2[0];
1933           g = bp2[1];
1934           b = bp2[2];
1935           dith = dmp[(x_align + x) & (DM_WIDTH - 1)] << 2;
1936           r1 = r + (dith >> r_prec);
1937           g1 = g + ((252 - dith) >> g_prec);
1938           b1 = b + (dith >> b_prec);
1939           pixel = (((r1 - (r1 >> r_prec)) >> r_right) << r_left) |
1940             (((g1 - (g1 >> g_prec)) >> g_right) << g_left) |
1941             (((b1 - (b1 >> b_prec)) >> b_right) << b_left);
1942           for (i = 0; i < bpp; i++)
1943             {
1944               *obptr++ = pixel & 0xff;
1945               pixel >>= 8;
1946             }
1947           bp2 += 3;
1948         }
1949       bptr += rowstride;
1950       obuf += bpl;
1951     }
1952 }
1953
1954 static void
1955 gdk_rgb_convert_truecolor_msb (GdkImage *image,
1956                                gint x0, gint y0, gint width, gint height,
1957                                guchar *buf, int rowstride,
1958                                gint x_align, gint y_align,
1959                                GdkRgbCmap *cmap)
1960 {
1961   int x, y;
1962   guchar *obuf, *obptr;
1963   gint bpl;
1964   guchar *bptr, *bp2;
1965   gint r, g, b;
1966   gint r_right, r_left;
1967   gint g_right, g_left;
1968   gint b_right, b_left;
1969   gint bpp;
1970   guint32 pixel;
1971   gint shift, shift_init;
1972
1973   r_right = 8 - image_info->visual->red_prec;
1974   r_left = image_info->visual->red_shift;
1975   g_right = 8 - image_info->visual->green_prec;
1976   g_left = image_info->visual->green_shift;
1977   b_right = 8 - image_info->visual->blue_prec;
1978   b_left = image_info->visual->blue_shift;
1979   bpp = image_info->bpp;
1980   bptr = buf;
1981   bpl = image->bpl;
1982   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * bpp;
1983   shift_init = (bpp - 1) << 3;
1984   for (y = 0; y < height; y++)
1985     {
1986       obptr = obuf;
1987       bp2 = bptr;
1988       for (x = 0; x < width; x++)
1989         {
1990           r = bp2[0];
1991           g = bp2[1];
1992           b = bp2[2];
1993           pixel = ((r >> r_right) << r_left) |
1994             ((g >> g_right) << g_left) |
1995             ((b >> b_right) << b_left);
1996           for (shift = shift_init; shift >= 0; shift -= 8)
1997             {
1998               *obptr++ = (pixel >> shift) & 0xff;
1999             }
2000           bp2 += 3;
2001         }
2002       bptr += rowstride;
2003       obuf += bpl;
2004     }
2005 }
2006
2007 static void
2008 gdk_rgb_convert_truecolor_msb_d (GdkImage *image,
2009                                  gint x0, gint y0, gint width, gint height,
2010                                  guchar *buf, int rowstride,
2011                                  gint x_align, gint y_align,
2012                                  GdkRgbCmap *cmap)
2013 {
2014   int x, y;
2015   guchar *obuf, *obptr;
2016   gint bpl;
2017   guchar *bptr, *bp2;
2018   gint r, g, b;
2019   gint r_right, r_left, r_prec;
2020   gint g_right, g_left, g_prec;
2021   gint b_right, b_left, b_prec;
2022   gint bpp;
2023   guint32 pixel;
2024   gint shift, shift_init;
2025   gint dith;
2026   gint r1, g1, b1;
2027   guchar *dmp;
2028
2029   r_right = 8 - image_info->visual->red_prec;
2030   r_left = image_info->visual->red_shift;
2031   r_prec = image_info->visual->red_prec;
2032   g_right = 8 - image_info->visual->green_prec;
2033   g_left = image_info->visual->green_shift;
2034   g_prec = image_info->visual->green_prec;
2035   b_right = 8 - image_info->visual->blue_prec;
2036   b_left = image_info->visual->blue_shift;
2037   b_prec = image_info->visual->blue_prec;
2038   bpp = image_info->bpp;
2039   bptr = buf;
2040   bpl = image->bpl;
2041   obuf = ((guchar *)image->mem) + y0 * bpl + x0 * bpp;
2042   shift_init = (bpp - 1) << 3;
2043   for (y = 0; y < height; y++)
2044     {
2045       dmp = DM[(y_align + y) & (DM_HEIGHT - 1)];
2046       obptr = obuf;
2047       bp2 = bptr;
2048       for (x = 0; x < width; x++)
2049         {
2050           r = bp2[0];
2051           g = bp2[1];
2052           b = bp2[2];
2053           dith = dmp[(x_align + x) & (DM_WIDTH - 1)] << 2;
2054           r1 = r + (dith >> r_prec);
2055           g1 = g + ((252 - dith) >> g_prec);
2056           b1 = b + (dith >> b_prec);
2057           pixel = (((r1 - (r1 >> r_prec)) >> r_right) << r_left) |
2058             (((g1 - (g1 >> g_prec)) >> g_right) << g_left) |
2059             (((b1 - (b1 >> b_prec)) >> b_right) << b_left);
2060           for (shift = shift_init; shift >= 0; shift -= 8)
2061             {
2062               *obptr++ = (pixel >> shift) & 0xff;
2063             }
2064           bp2 += 3;
2065         }
2066       bptr += rowstride;
2067       obuf += bpl;
2068     }
2069 }
2070
2071 /* Returns a pointer to the stage buffer. */
2072 static guchar *
2073 gdk_rgb_ensure_stage (void)
2074 {
2075   if (image_info->stage_buf == NULL)
2076     image_info->stage_buf = g_malloc (IMAGE_HEIGHT * STAGE_ROWSTRIDE);
2077   return image_info->stage_buf;
2078 }
2079
2080 /* This is slow. Speed me up, please. */
2081 static void
2082 gdk_rgb_32_to_stage (guchar *buf, gint rowstride, gint width, gint height)
2083 {
2084   gint x, y;
2085   guchar *pi_start, *po_start;
2086   guchar *pi, *po;
2087
2088   pi_start = buf;
2089   po_start = gdk_rgb_ensure_stage ();
2090   for (y = 0; y < height; y++)
2091     {
2092       pi = pi_start;
2093       po = po_start;
2094       for (x = 0; x < width; x++)
2095         {
2096           *po++ = *pi++;
2097           *po++ = *pi++;
2098           *po++ = *pi++;
2099           pi++;
2100         }
2101       pi_start += rowstride;
2102       po_start += STAGE_ROWSTRIDE;
2103     }
2104 }
2105
2106 /* Generic 32bit RGB conversion function - convert to 24bit packed, then
2107    go from there. */
2108 static void
2109 gdk_rgb_convert_32_generic (GdkImage *image,
2110                             gint x0, gint y0, gint width, gint height,
2111                             guchar *buf, gint rowstride,
2112                             gint x_align, gint y_align, GdkRgbCmap *cmap)
2113 {
2114   gdk_rgb_32_to_stage (buf, rowstride, width, height);
2115
2116   (*image_info->conv) (image, x0, y0, width, height,
2117                        image_info->stage_buf, STAGE_ROWSTRIDE,
2118                        x_align, y_align, cmap);
2119 }
2120
2121 /* Generic 32bit RGB conversion function - convert to 24bit packed, then
2122    go from there. */
2123 static void
2124 gdk_rgb_convert_32_generic_d (GdkImage *image,
2125                               gint x0, gint y0, gint width, gint height,
2126                               guchar *buf, gint rowstride,
2127                               gint x_align, gint y_align, GdkRgbCmap *cmap)
2128 {
2129   gdk_rgb_32_to_stage (buf, rowstride, width, height);
2130
2131   (*image_info->conv_d) (image, x0, y0, width, height,
2132                          image_info->stage_buf, STAGE_ROWSTRIDE,
2133                          x_align, y_align, cmap);
2134 }
2135
2136 /* This is slow. Speed me up, please. */
2137 static void
2138 gdk_rgb_gray_to_stage (guchar *buf, gint rowstride, gint width, gint height)
2139 {
2140   gint x, y;
2141   guchar *pi_start, *po_start;
2142   guchar *pi, *po;
2143   guchar gray;
2144
2145   pi_start = buf;
2146   po_start = gdk_rgb_ensure_stage ();
2147   for (y = 0; y < height; y++)
2148     {
2149       pi = pi_start;
2150       po = po_start;
2151       for (x = 0; x < width; x++)
2152         {
2153           gray = *pi++;
2154           *po++ = gray;
2155           *po++ = gray;
2156           *po++ = gray;
2157         }
2158       pi_start += rowstride;
2159       po_start += STAGE_ROWSTRIDE;
2160     }
2161 }
2162
2163 /* Generic gray conversion function - convert to 24bit packed, then go
2164    from there. */
2165 static void
2166 gdk_rgb_convert_gray_generic (GdkImage *image,
2167                               gint x0, gint y0, gint width, gint height,
2168                               guchar *buf, gint rowstride,
2169                               gint x_align, gint y_align, GdkRgbCmap *cmap)
2170 {
2171   gdk_rgb_gray_to_stage (buf, rowstride, width, height);
2172
2173   (*image_info->conv) (image, x0, y0, width, height,
2174                        image_info->stage_buf, STAGE_ROWSTRIDE,
2175                        x_align, y_align, cmap);
2176 }
2177
2178 static void
2179 gdk_rgb_convert_gray_generic_d (GdkImage *image,
2180                                 gint x0, gint y0, gint width, gint height,
2181                                 guchar *buf, gint rowstride,
2182                                 gint x_align, gint y_align, GdkRgbCmap *cmap)
2183 {
2184   gdk_rgb_gray_to_stage (buf, rowstride, width, height);
2185
2186   (*image_info->conv_d) (image, x0, y0, width, height,
2187                          image_info->stage_buf, STAGE_ROWSTRIDE,
2188                          x_align, y_align, cmap);
2189 }
2190
2191 /* Render grayscale using indexed method. */
2192 static void
2193 gdk_rgb_convert_gray_cmap (GdkImage *image,
2194                            gint x0, gint y0, gint width, gint height,
2195                            guchar *buf, gint rowstride,
2196                            gint x_align, gint y_align, GdkRgbCmap *cmap)
2197 {
2198   (*image_info->conv_indexed) (image, x0, y0, width, height,
2199                                buf, rowstride,
2200                                x_align, y_align, image_info->gray_cmap);
2201 }
2202
2203 #if 0
2204 static void
2205 gdk_rgb_convert_gray_cmap_d (GdkImage *image,
2206                                 gint x0, gint y0, gint width, gint height,
2207                                 guchar *buf, gint rowstride,
2208                                 gint x_align, gint y_align, GdkRgbCmap *cmap)
2209 {
2210   (*image_info->conv_indexed_d) (image, x0, y0, width, height,
2211                                  buf, rowstride,
2212                                  x_align, y_align, image_info->gray_cmap);
2213 }
2214 #endif
2215
2216 /* This is slow. Speed me up, please. */
2217 static void
2218 gdk_rgb_indexed_to_stage (guchar *buf, gint rowstride, gint width, gint height,
2219                           GdkRgbCmap *cmap)
2220 {
2221   gint x, y;
2222   guchar *pi_start, *po_start;
2223   guchar *pi, *po;
2224   gint rgb;
2225
2226   pi_start = buf;
2227   po_start = gdk_rgb_ensure_stage ();
2228   for (y = 0; y < height; y++)
2229     {
2230       pi = pi_start;
2231       po = po_start;
2232       for (x = 0; x < width; x++)
2233         {
2234           rgb = cmap->colors[*pi++];
2235           *po++ = rgb >> 16;
2236           *po++ = (rgb >> 8) & 0xff;
2237           *po++ = rgb & 0xff;
2238         }
2239       pi_start += rowstride;
2240       po_start += STAGE_ROWSTRIDE;
2241     }
2242 }
2243
2244 /* Generic gray conversion function - convert to 24bit packed, then go
2245    from there. */
2246 static void
2247 gdk_rgb_convert_indexed_generic (GdkImage *image,
2248                                  gint x0, gint y0, gint width, gint height,
2249                                  guchar *buf, gint rowstride,
2250                                  gint x_align, gint y_align, GdkRgbCmap *cmap)
2251 {
2252   gdk_rgb_indexed_to_stage (buf, rowstride, width, height, cmap);
2253
2254   (*image_info->conv) (image, x0, y0, width, height,
2255                        image_info->stage_buf, STAGE_ROWSTRIDE,
2256                        x_align, y_align, cmap);
2257 }
2258
2259 static void
2260 gdk_rgb_convert_indexed_generic_d (GdkImage *image,
2261                                    gint x0, gint y0, gint width, gint height,
2262                                    guchar *buf, gint rowstride,
2263                                    gint x_align, gint y_align,
2264                                    GdkRgbCmap *cmap)
2265 {
2266   gdk_rgb_indexed_to_stage (buf, rowstride, width, height, cmap);
2267
2268   (*image_info->conv_d) (image, x0, y0, width, height,
2269                          image_info->stage_buf, STAGE_ROWSTRIDE,
2270                          x_align, y_align, cmap);
2271 }
2272
2273 /* Select a conversion function based on the visual and a
2274    representative image. */
2275 static void
2276 gdk_rgb_select_conv (GdkImage *image)
2277 {
2278   GdkByteOrder byte_order;
2279   gint depth, bpp, byterev;
2280   GdkVisualType vtype;
2281   guint32 red_mask, green_mask, blue_mask;
2282   GdkRgbConvFunc conv, conv_d;
2283   GdkRgbConvFunc conv_32, conv_32_d;
2284   GdkRgbConvFunc conv_gray, conv_gray_d;
2285   GdkRgbConvFunc conv_indexed, conv_indexed_d;
2286
2287   depth = image_info->visual->depth;
2288   bpp = image->bpp;
2289
2290   byte_order = image->byte_order;
2291
2292 #ifdef WORDS_BIGENDIAN
2293   byterev = (byte_order == GDK_LSB_FIRST);
2294 #else
2295   byterev = (byte_order == GDK_MSB_FIRST);
2296 #endif
2297
2298   vtype = image_info->visual->type;
2299   if (vtype == GDK_VISUAL_DIRECT_COLOR)
2300     vtype = GDK_VISUAL_TRUE_COLOR;
2301
2302   red_mask = image_info->visual->red_mask;
2303   green_mask = image_info->visual->green_mask;
2304   blue_mask = image_info->visual->blue_mask;
2305
2306   conv = NULL;
2307   conv_d = NULL;
2308
2309   conv_32 = gdk_rgb_convert_32_generic;
2310   conv_32_d = gdk_rgb_convert_32_generic_d;
2311
2312   conv_gray = gdk_rgb_convert_gray_generic;
2313   conv_gray_d = gdk_rgb_convert_gray_generic_d;
2314
2315   conv_indexed = gdk_rgb_convert_indexed_generic;
2316   conv_indexed_d = gdk_rgb_convert_indexed_generic_d;
2317
2318   image_info->dith_default = FALSE;
2319
2320   if (bpp == 2 && depth == 16 && !byterev &&
2321       red_mask == 0xf800 && green_mask == 0x7e0 && blue_mask == 0x1f)
2322     {
2323       conv = gdk_rgb_convert_565;
2324       conv_d = gdk_rgb_convert_565_d;
2325       conv_gray = gdk_rgb_convert_565_gray;
2326       gdk_rgb_preprocess_dm_565 ();
2327     }
2328   else if (bpp == 2 && depth == 16 &&
2329            vtype == GDK_VISUAL_TRUE_COLOR && byterev &&
2330       red_mask == 0xf800 && green_mask == 0x7e0 && blue_mask == 0x1f)
2331     conv = gdk_rgb_convert_565_br;
2332
2333   else if (bpp == 2 && depth == 15 &&
2334            vtype == GDK_VISUAL_TRUE_COLOR && !byterev &&
2335       red_mask == 0x7c00 && green_mask == 0x3e0 && blue_mask == 0x1f)
2336     conv = gdk_rgb_convert_555;
2337
2338   else if (bpp == 2 && depth == 15 &&
2339            vtype == GDK_VISUAL_TRUE_COLOR && byterev &&
2340       red_mask == 0x7c00 && green_mask == 0x3e0 && blue_mask == 0x1f)
2341     conv = gdk_rgb_convert_555_br;
2342
2343   /* I'm not 100% sure about the 24bpp tests */
2344   else if (bpp == 3 && depth == 24 &&
2345            vtype == GDK_VISUAL_TRUE_COLOR && byte_order == GDK_LSB_FIRST &&
2346            red_mask == 0xff0000 && green_mask == 0xff00 && blue_mask == 0xff)
2347     conv = gdk_rgb_convert_888_lsb;
2348   else if (bpp == 3 && depth == 24 &&
2349            vtype == GDK_VISUAL_TRUE_COLOR && byte_order == GDK_MSB_FIRST &&
2350            red_mask == 0xff0000 && green_mask == 0xff00 && blue_mask == 0xff)
2351     conv = gdk_rgb_convert_888_msb;
2352   else if (bpp == 4 && depth == 24 &&
2353            vtype == GDK_VISUAL_TRUE_COLOR && !byterev &&
2354            red_mask == 0xff0000 && green_mask == 0xff00 && blue_mask == 0xff)
2355     conv = gdk_rgb_convert_0888;
2356   else if (bpp == 4 && depth == 24 &&
2357            vtype == GDK_VISUAL_TRUE_COLOR && byterev &&
2358            red_mask == 0xff0000 && green_mask == 0xff00 && blue_mask == 0xff)
2359     conv = gdk_rgb_convert_0888_br;
2360
2361   else if (vtype == GDK_VISUAL_TRUE_COLOR && byte_order == GDK_LSB_FIRST)
2362     {
2363       conv = gdk_rgb_convert_truecolor_lsb;
2364       conv_d = gdk_rgb_convert_truecolor_lsb_d;
2365     }
2366   else if (vtype == GDK_VISUAL_TRUE_COLOR && byte_order == GDK_MSB_FIRST)
2367     {
2368       conv = gdk_rgb_convert_truecolor_msb;
2369       conv = gdk_rgb_convert_truecolor_msb_d;
2370     }
2371   else if (bpp == 1 && depth == 8 && (vtype == GDK_VISUAL_PSEUDO_COLOR
2372 #ifdef ENABLE_GRAYSCALE
2373                                       || vtype == GDK_VISUAL_GRAYSCALE
2374 #endif
2375                                       ))
2376     {
2377       image_info->dith_default = TRUE;
2378       conv = gdk_rgb_convert_8;
2379       if (vtype != GDK_VISUAL_GRAYSCALE)
2380         {
2381           if (image_info->nred_shades == 6 &&
2382               image_info->ngreen_shades == 6 &&
2383               image_info->nblue_shades == 6)
2384             conv_d = gdk_rgb_convert_8_d666;
2385           else
2386             conv_d = gdk_rgb_convert_8_d;
2387         }
2388       conv_indexed = gdk_rgb_convert_8_indexed;
2389       conv_gray = gdk_rgb_convert_gray_cmap;
2390     }
2391   else if (bpp == 1 && depth == 8 && (vtype == GDK_VISUAL_STATIC_GRAY
2392 #ifdef not_ENABLE_GRAYSCALE
2393                                       || vtype == GDK_VISUAL_GRAYSCALE
2394 #endif
2395                                       ))
2396     {
2397       conv = gdk_rgb_convert_gray8;
2398       conv_gray = gdk_rgb_convert_gray8_gray;
2399     }
2400
2401   if (conv_d == NULL)
2402     conv_d = conv;
2403
2404   image_info->conv = conv;
2405   image_info->conv_d = conv_d;
2406
2407   image_info->conv_32 = conv_32;
2408   image_info->conv_32_d = conv_32_d;
2409
2410   image_info->conv_gray = conv_gray;
2411   image_info->conv_gray_d = conv_gray_d;
2412
2413   image_info->conv_indexed = conv_indexed;
2414   image_info->conv_indexed_d = conv_indexed_d;
2415 }
2416
2417 static gint horiz_idx;
2418 static gint horiz_y = IMAGE_HEIGHT;
2419 static gint vert_idx;
2420 static gint vert_x = IMAGE_WIDTH;
2421 static gint tile_idx;
2422 static gint tile_x = IMAGE_WIDTH;
2423 static gint tile_y1 = IMAGE_HEIGHT;
2424 static gint tile_y2 = IMAGE_HEIGHT;
2425
2426 #ifdef VERBOSE
2427 static gint sincelast;
2428 #endif
2429
2430 /* Defining NO_FLUSH can cause inconsistent screen updates, but is useful
2431    for performance evaluation. */
2432
2433 #undef NO_FLUSH
2434
2435 static gint
2436 gdk_rgb_alloc_scratch_image (void)
2437 {
2438   if (static_image_idx == N_IMAGES)
2439     {
2440 #ifndef NO_FLUSH
2441       gdk_flush ();
2442 #endif
2443 #ifdef VERBOSE
2444       g_print ("flush, %d puts since last flush\n", sincelast);
2445       sincelast = 0;
2446 #endif
2447       static_image_idx = 0;
2448       horiz_y = IMAGE_HEIGHT;
2449       vert_x = IMAGE_WIDTH;
2450       tile_x = IMAGE_WIDTH;
2451       tile_y1 = tile_y2 = IMAGE_HEIGHT;
2452     }
2453   return static_image_idx++;
2454 }
2455
2456 static GdkImage *
2457 gdk_rgb_alloc_scratch (gint width, gint height, gint *x0, gint *y0)
2458 {
2459   GdkImage *image;
2460   gint idx;
2461
2462
2463   if (width >= (IMAGE_WIDTH >> 1))
2464     {
2465       if (height >= (IMAGE_HEIGHT >> 1))
2466         {
2467           idx = gdk_rgb_alloc_scratch_image ();
2468           *x0 = 0;
2469           *y0 = 0;
2470         }
2471       else
2472         {
2473           if (height + horiz_y > IMAGE_HEIGHT)
2474             {
2475               horiz_idx = gdk_rgb_alloc_scratch_image ();
2476               horiz_y = 0;
2477             }
2478           idx = horiz_idx;
2479           *x0 = 0;
2480           *y0 = horiz_y;
2481           horiz_y += height;
2482         }
2483     }
2484   else
2485     {
2486       if (height >= (IMAGE_HEIGHT >> 1))
2487         {
2488           if (width + vert_x > IMAGE_WIDTH)
2489             {
2490               vert_idx = gdk_rgb_alloc_scratch_image ();
2491               vert_x = 0;
2492             }
2493           idx = vert_idx;
2494           *x0 = vert_x;
2495           *y0 = 0;
2496           vert_x += (width + 3) & -4;
2497         }
2498       else
2499         {
2500           if (width + tile_x > IMAGE_WIDTH)
2501             {
2502               tile_y1 = tile_y2;
2503               tile_x = 0;
2504             }
2505           if (height + tile_y1 > IMAGE_HEIGHT)
2506             {
2507               tile_idx = gdk_rgb_alloc_scratch_image ();
2508               tile_x = 0;
2509               tile_y1 = 0;
2510               tile_y2 = 0;
2511             }
2512           if (height + tile_y1 > tile_y2)
2513             tile_y2 = height + tile_y1;
2514           idx = tile_idx;
2515           *x0 = tile_x;
2516           *y0 = tile_y1;
2517           tile_x += (width + 3) & -4;
2518         }
2519     }
2520   image = static_image[idx];
2521 #ifdef VERBOSE
2522   g_print ("index %d, x %d, y %d (%d x %d)\n", idx, *x0, *y0, width, height);
2523   sincelast++;
2524 #endif
2525   return image;
2526 }
2527
2528 static void
2529 gdk_draw_rgb_image_core (GdkDrawable *drawable,
2530                          GdkGC *gc,
2531                          gint x,
2532                          gint y,
2533                          gint width,
2534                          gint height,
2535                          guchar *buf,
2536                          gint pixstride,
2537                          gint rowstride,
2538                          GdkRgbConvFunc conv,
2539                          GdkRgbCmap *cmap)
2540 {
2541   gint y0, x0;
2542   gint xs0, ys0;
2543   GdkImage *image;
2544   gint width1, height1;
2545   guchar *buf_ptr;
2546
2547   for (y0 = 0; y0 < height; y0 += IMAGE_HEIGHT)
2548     {
2549       height1 = MIN (height - y0, IMAGE_HEIGHT);
2550       for (x0 = 0; x0 < width; x0 += IMAGE_WIDTH)
2551         {
2552           width1 = MIN (width - x0, IMAGE_WIDTH);
2553           buf_ptr = buf + y0 * rowstride + x0 * pixstride;
2554
2555           image = gdk_rgb_alloc_scratch (width1, height1, &xs0, &ys0);
2556
2557           conv (image, xs0, ys0, width1, height1, buf_ptr, rowstride,
2558                 x + x0, y + y0, cmap);
2559
2560 #ifndef DONT_ACTUALLY_DRAW
2561           gdk_draw_image (drawable, gc,
2562                           image, xs0, ys0, x + x0, y + y0, width1, height1);
2563 #endif
2564         }
2565     }
2566 }
2567
2568
2569 void
2570 gdk_draw_rgb_image (GdkDrawable *drawable,
2571                     GdkGC *gc,
2572                     gint x,
2573                     gint y,
2574                     gint width,
2575                     gint height,
2576                     GdkRgbDither dith,
2577                     guchar *rgb_buf,
2578                     gint rowstride)
2579 {
2580   if (dith == GDK_RGB_DITHER_NONE || (dith == GDK_RGB_DITHER_NORMAL &&
2581                                       !image_info->dith_default))
2582     gdk_draw_rgb_image_core (drawable, gc, x, y, width, height,
2583                              rgb_buf, 3, rowstride, image_info->conv, NULL);
2584   else
2585     gdk_draw_rgb_image_core (drawable, gc, x, y, width, height,
2586                              rgb_buf, 3, rowstride, image_info->conv_d, NULL);
2587 }
2588
2589 void
2590 gdk_draw_rgb_32_image (GdkDrawable *drawable,
2591                        GdkGC *gc,
2592                        gint x,
2593                        gint y,
2594                        gint width,
2595                        gint height,
2596                        GdkRgbDither dith,
2597                        guchar *buf,
2598                        gint rowstride)
2599 {
2600   if (dith == GDK_RGB_DITHER_NONE || (dith == GDK_RGB_DITHER_NORMAL &&
2601                                       !image_info->dith_default))
2602     gdk_draw_rgb_image_core (drawable, gc, x, y, width, height,
2603                              buf, 4, rowstride,
2604                              image_info->conv_32, NULL);
2605   else
2606     gdk_draw_rgb_image_core (drawable, gc, x, y, width, height,
2607                              buf, 4, rowstride,
2608                              image_info->conv_32_d, NULL);
2609 }
2610
2611 static void
2612 gdk_rgb_make_gray_cmap (GdkRgbInfo *info)
2613 {
2614   guint32 rgb[256];
2615   gint i;
2616
2617   for (i = 0; i < 256; i++)
2618     rgb[i] = (i << 16)  | (i << 8) | i;
2619   info->gray_cmap = gdk_rgb_cmap_new (rgb, 256);
2620 }
2621
2622 void
2623 gdk_draw_gray_image (GdkDrawable *drawable,
2624                      GdkGC *gc,
2625                      gint x,
2626                      gint y,
2627                      gint width,
2628                      gint height,
2629                      GdkRgbDither dith,
2630                      guchar *buf,
2631                      gint rowstride)
2632 {
2633   if (image_info->bpp == 1 &&
2634       image_info->gray_cmap == NULL &&
2635       (image_info->visual->type == GDK_VISUAL_PSEUDO_COLOR ||
2636        image_info->visual->type == GDK_VISUAL_GRAYSCALE))
2637     gdk_rgb_make_gray_cmap (image_info);
2638   
2639   if (dith == GDK_RGB_DITHER_NONE || (dith == GDK_RGB_DITHER_NORMAL &&
2640                                       !image_info->dith_default))
2641     gdk_draw_rgb_image_core (drawable, gc, x, y, width, height,
2642                              buf, 1, rowstride,
2643                              image_info->conv_gray, NULL);
2644   else
2645     gdk_draw_rgb_image_core (drawable, gc, x, y, width, height,
2646                              buf, 1, rowstride,
2647                              image_info->conv_gray_d, NULL);
2648 }
2649
2650 GdkRgbCmap *
2651 gdk_rgb_cmap_new (guint32 *colors, gint n_colors)
2652 {
2653   GdkRgbCmap *cmap;
2654   int i, j;
2655   guint32 rgb;
2656
2657   g_return_val_if_fail (n_colors >= 0, NULL);
2658   g_return_val_if_fail (n_colors <= 256, NULL);
2659   cmap = g_new (GdkRgbCmap, 1);
2660   memcpy (cmap->colors, colors, n_colors * sizeof(guint32));
2661   if (image_info->bpp == 1 &&
2662       (image_info->visual->type == GDK_VISUAL_PSEUDO_COLOR ||
2663        image_info->visual->type == GDK_VISUAL_GRAYSCALE))
2664     for (i = 0; i < n_colors; i++)
2665       {
2666         rgb = colors[i];
2667         j = ((rgb & 0xf00000) >> 12) |
2668                    ((rgb & 0xf000) >> 8) |
2669                    ((rgb & 0xf0) >> 4);
2670 #ifdef VERBOSE
2671         g_print ("%d %x %x %d\n", i, j, colorcube[j]);
2672 #endif
2673         cmap->lut[i] = colorcube[j];
2674       }
2675   return cmap;
2676 }
2677
2678 void
2679 gdk_rgb_cmap_free (GdkRgbCmap *cmap)
2680 {
2681   g_free (cmap);
2682 }
2683
2684 void
2685 gdk_draw_indexed_image (GdkDrawable *drawable,
2686                         GdkGC *gc,
2687                         gint x,
2688                         gint y,
2689                         gint width,
2690                         gint height,
2691                         GdkRgbDither dith,
2692                         guchar *buf,
2693                         gint rowstride,
2694                         GdkRgbCmap *cmap)
2695 {
2696   if (dith == GDK_RGB_DITHER_NONE || (dith == GDK_RGB_DITHER_NORMAL &&
2697                                       !image_info->dith_default))
2698     gdk_draw_rgb_image_core (drawable, gc, x, y, width, height,
2699                              buf, 1, rowstride,
2700                              image_info->conv_indexed, cmap);
2701   else
2702     gdk_draw_rgb_image_core (drawable, gc, x, y, width, height,
2703                              buf, 1, rowstride,
2704                              image_info->conv_indexed_d, cmap);
2705 }
2706
2707 gboolean
2708 gdk_rgb_ditherable (void)
2709 {
2710   return (image_info->conv != image_info->conv_d);
2711 }
2712
2713 GdkColormap *
2714 gdk_rgb_get_cmap (void)
2715 {
2716   return image_info->cmap;
2717 }
2718
2719 GdkVisual *
2720 gdk_rgb_get_visual (void)
2721 {
2722   return image_info->visual;
2723 }