/[rdesktop]/sourceforge.net/trunk/rdesktop/xwin.c
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Diff of /sourceforge.net/trunk/rdesktop/xwin.c

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 28 by matty, Wed Jun 20 13:54:48 2001 UTC revision 288 by jsorg71, Sun Jan 5 20:06:32 2003 UTC
# Line 1  Line 1 
1  /*  /*
2     rdesktop: A Remote Desktop Protocol client.     rdesktop: A Remote Desktop Protocol client.
3     User interface services - X-Windows     User interface services - X Window System
4     Copyright (C) Matthew Chapman 1999-2000     Copyright (C) Matthew Chapman 1999-2002
5      
6     This program is free software; you can redistribute it and/or modify     This program is free software; you can redistribute it and/or modify
7     it under the terms of the GNU General Public License as published by     it under the terms of the GNU General Public License as published by
8     the Free Software Foundation; either version 2 of the License, or     the Free Software Foundation; either version 2 of the License, or
9     (at your option) any later version.     (at your option) any later version.
10      
11     This program is distributed in the hope that it will be useful,     This program is distributed in the hope that it will be useful,
12     but WITHOUT ANY WARRANTY; without even the implied warranty of     but WITHOUT ANY WARRANTY; without even the implied warranty of
13     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14     GNU General Public License for more details.     GNU General Public License for more details.
15      
16     You should have received a copy of the GNU General Public License     You should have received a copy of the GNU General Public License
17     along with this program; if not, write to the Free Software     along with this program; if not, write to the Free Software
18     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
# Line 21  Line 21 
21  #include <X11/Xlib.h>  #include <X11/Xlib.h>
22  #include <X11/Xutil.h>  #include <X11/Xutil.h>
23  #include <time.h>  #include <time.h>
24    #include <errno.h>
25  #include "rdesktop.h"  #include "rdesktop.h"
26    
27  extern int width;  extern int width;
28  extern int height;  extern int height;
29  extern BOOL motion;  extern BOOL sendmotion;
 extern BOOL grab_keyboard;  
30  extern BOOL fullscreen;  extern BOOL fullscreen;
31  extern int private_colormap;  extern BOOL grab_keyboard;
32    extern BOOL hide_decorations;
33  static int bpp;  extern char title[];
34  static int depth;  BOOL enable_compose = False;
35  static Display *display;  BOOL focused;
36    BOOL mouse_in_wnd;
37    
38    Display *display;
39    static int x_socket;
40    static Screen *screen;
41  static Window wnd;  static Window wnd;
42  static GC gc;  static GC gc;
43  static Visual *visual;  static Visual *visual;
44  static uint32 *colmap;  static int depth;
45    static int bpp;
46    static XIM IM;
47    static XIC IC;
48    static XModifierKeymap *mod_map;
49    static Cursor current_cursor;
50    static Atom protocol_atom, kill_atom;
51    
52    /* endianness */
53    static BOOL host_be;
54    static BOOL xserver_be;
55    
56    /* software backing store */
57    static BOOL ownbackstore;
58    static Pixmap backstore;
59    
60    /* MWM decorations */
61    #define MWM_HINTS_DECORATIONS   (1L << 1)
62    #define PROP_MOTIF_WM_HINTS_ELEMENTS    5
63    typedef struct
64    {
65            unsigned long flags;
66            unsigned long functions;
67            unsigned long decorations;
68            long inputMode;
69            unsigned long status;
70    }
71    PropMotifWmHints;
72    
 #define Ctrans(col) ( private_colormap ? col : colmap[col])  
73    
74  #define L_ENDIAN  #define FILL_RECTANGLE(x,y,cx,cy)\
75  int screen_msbfirst = 0;  { \
76            XFillRectangle(display, wnd, gc, x, y, cx, cy); \
77            if (ownbackstore) \
78                    XFillRectangle(display, backstore, gc, x, y, cx, cy); \
79    }
80    
81    #define FILL_RECTANGLE_FAST(x,y,cx,cy)\
82    { \
83            XFillRectangle(display, ownbackstore ? backstore : wnd, gc, x, y, cx, cy); \
84    }
85    
86    /* colour maps */
87    BOOL owncolmap = False;
88    static Colormap xcolmap;
89    static uint32 *colmap;
90    
91  static uint8 *translate(int width, int height, uint8 *data);  #define TRANSLATE(col)          ( owncolmap ? col : translate_colour(colmap[col]) )
92    #define SET_FOREGROUND(col)     XSetForeground(display, gc, TRANSLATE(col));
93    #define SET_BACKGROUND(col)     XSetBackground(display, gc, TRANSLATE(col));
94    
95  static int rop2_map[] = {  static int rop2_map[] = {
96          GXclear,                /* 0 */          GXclear,                /* 0 */
# Line 64  static int rop2_map[] = { Line 111  static int rop2_map[] = {
111          GXset                   /* 1 */          GXset                   /* 1 */
112  };  };
113    
114  static void  #define SET_FUNCTION(rop2)      { if (rop2 != ROP2_COPY) XSetFunction(display, gc, rop2_map[rop2]); }
115  xwin_set_function(uint8 rop2)  #define RESET_FUNCTION(rop2)    { if (rop2 != ROP2_COPY) XSetFunction(display, gc, GXcopy); }
116    
117    void
118    mwm_hide_decorations(void)
119  {  {
120          static uint8 last_rop2 = ROP2_COPY;          PropMotifWmHints motif_hints;
121            Atom hintsatom;
122    
123            /* setup the property */
124            motif_hints.flags = MWM_HINTS_DECORATIONS;
125            motif_hints.decorations = 0;
126    
127          if (last_rop2 != rop2)          /* get the atom for the property */
128            hintsatom = XInternAtom(display, "_MOTIF_WM_HINTS", False);
129            if (!hintsatom)
130          {          {
131                  XSetFunction(display, gc, rop2_map[rop2]);                  error("Failed to get atom _MOTIF_WM_HINTS\n");
132                  last_rop2 = rop2;                  return;
133          }          }
134    
135            XChangeProperty(display, wnd, hintsatom, hintsatom, 32, PropModeReplace,
136                            (unsigned char *) &motif_hints, PROP_MOTIF_WM_HINTS_ELEMENTS);
137    }
138    
139    static void
140    translate8(uint8 * data, uint8 * out, uint8 * end)
141    {
142            while (out < end)
143                    *(out++) = (uint8) colmap[*(data++)];
144    }
145    
146    static void
147    translate16(uint8 * data, uint16 * out, uint16 * end)
148    {
149            while (out < end)
150                    *(out++) = (uint16) colmap[*(data++)];
151  }  }
152    
153    /* little endian - conversion happens when colourmap is built */
154  static void  static void
155  xwin_grab_keyboard()  translate24(uint8 * data, uint8 * out, uint8 * end)
156  {  {
157          XGrabKeyboard(display, wnd, True, GrabModeAsync, GrabModeAsync,          uint32 value;
158                        CurrentTime);  
159            while (out < end)
160            {
161                    value = colmap[*(data++)];
162                    *(out++) = value;
163                    *(out++) = value >> 8;
164                    *(out++) = value >> 16;
165            }
166  }  }
167    
168  static void  static void
169  xwin_ungrab_keyboard()  translate32(uint8 * data, uint32 * out, uint32 * end)
170    {
171            while (out < end)
172                    *(out++) = colmap[*(data++)];
173    }
174    
175    static uint8 *
176    translate_image(int width, int height, uint8 * data)
177  {  {
178          XUngrabKeyboard(display, CurrentTime);          int size = width * height * bpp / 8;
179            uint8 *out = xmalloc(size);
180            uint8 *end = out + size;
181    
182            switch (bpp)
183            {
184                    case 8:
185                            translate8(data, out, end);
186                            break;
187    
188                    case 16:
189                            translate16(data, (uint16 *) out, (uint16 *) end);
190                            break;
191    
192                    case 24:
193                            translate24(data, out, end);
194                            break;
195    
196                    case 32:
197                            translate32(data, (uint32 *) out, (uint32 *) end);
198                            break;
199            }
200    
201            return out;
202    }
203    
204    #define BSWAP16(x) { x = (((x & 0xff) << 8) | (x >> 8)); }
205    #define BSWAP24(x) { x = (((x & 0xff) << 16) | (x >> 16) | ((x >> 8) & 0xff00)); }
206    #define BSWAP32(x) { x = (((x & 0xff00ff) << 8) | ((x >> 8) & 0xff00ff)); \
207                            x = (x << 16) | (x >> 16); }
208    
209    static uint32
210    translate_colour(uint32 colour)
211    {
212            switch (bpp)
213            {
214                    case 16:
215                            if (host_be != xserver_be)
216                                    BSWAP16(colour);
217                            break;
218    
219                    case 24:
220                            if (xserver_be)
221                                    BSWAP24(colour);
222                            break;
223    
224                    case 32:
225                            if (host_be != xserver_be)
226                                    BSWAP32(colour);
227                            break;
228            }
229    
230            return colour;
231  }  }
232    
233  BOOL  BOOL
234  ui_create_window(char *title)  get_key_state(unsigned int state, uint32 keysym)
235    {
236            int modifierpos, key, keysymMask = 0;
237            int offset;
238    
239            KeyCode keycode = XKeysymToKeycode(display, keysym);
240    
241            if (keycode == NoSymbol)
242                    return False;
243    
244            for (modifierpos = 0; modifierpos < 8; modifierpos++)
245            {
246                    offset = mod_map->max_keypermod * modifierpos;
247    
248                    for (key = 0; key < mod_map->max_keypermod; key++)
249                    {
250                            if (mod_map->modifiermap[offset + key] == keycode)
251                                    keysymMask |= 1 << modifierpos;
252                    }
253            }
254    
255            return (state & keysymMask) ? True : False;
256    }
257    
258    BOOL
259    ui_init(void)
260  {  {
         XSetWindowAttributes attribs;  
         XClassHint *classhints;  
         XSizeHints *sizehints;  
         unsigned long input_mask;  
261          XPixmapFormatValues *pfm;          XPixmapFormatValues *pfm;
262          int count;          uint16 test;
263            int i;
264    
265          display = XOpenDisplay(NULL);          display = XOpenDisplay(NULL);
266          if (display == NULL)          if (display == NULL)
267          {          {
268                  ERROR("Failed to open display\n");                  error("Failed to open display: %s\n", XDisplayName(NULL));
269                  return False;                  return False;
270          }          }
271    
272          visual = DefaultVisual(display, DefaultScreen(display));          x_socket = ConnectionNumber(display);
273          depth = DefaultDepth(display, DefaultScreen(display));          screen = DefaultScreenOfDisplay(display);
274          pfm = XListPixmapFormats(display, &count);          visual = DefaultVisualOfScreen(screen);
275            depth = DefaultDepthOfScreen(screen);
276    
277            pfm = XListPixmapFormats(display, &i);
278          if (pfm != NULL)          if (pfm != NULL)
279          {          {
280                  while (count--)                  /* Use maximum bpp for this depth - this is generally
281                       desirable, e.g. 24 bits->32 bits. */
282                    while (i--)
283                  {                  {
284                          if ((pfm + count)->depth == depth                          if ((pfm[i].depth == depth) && (pfm[i].bits_per_pixel > bpp))
                             && (pfm + count)->bits_per_pixel > bpp)  
285                          {                          {
286                                  bpp = (pfm + count)->bits_per_pixel;                                  bpp = pfm[i].bits_per_pixel;
287                          }                          }
288                  }                  }
289                  XFree(pfm);                  XFree(pfm);
# Line 124  ui_create_window(char *title) Line 291  ui_create_window(char *title)
291    
292          if (bpp < 8)          if (bpp < 8)
293          {          {
294                  ERROR("Less than 8 bpp not currently supported.\n");                  error("Less than 8 bpp not currently supported.\n");
295                  XCloseDisplay(display);                  XCloseDisplay(display);
296                  return False;                  return False;
297          }          }
298    
299          width &= ~3; /* make width nicely divisible */          if (owncolmap != True)
300            {
301                    xcolmap = DefaultColormapOfScreen(screen);
302                    if (depth <= 8)
303                    {
304                            printf("You're using a screen depth of 8-bits or lower\n");
305                            printf("If you get scewed colours, try the -C switch\n");
306                    }
307            }
308    
309            gc = XCreateGC(display, RootWindowOfScreen(screen), 0, NULL);
310    
311            if (DoesBackingStore(screen) != Always)
312                    ownbackstore = True;
313    
314            test = 1;
315            host_be = !(BOOL) (*(uint8 *) (&test));
316            xserver_be = (ImageByteOrder(display) == MSBFirst);
317    
318            if ((width == 0) || (height == 0))
319            {
320                    /* Fetch geometry from _NET_WORKAREA */
321                    uint32 xpos, ypos;
322    
323          attribs.background_pixel = BlackPixel(display, DefaultScreen(display));                  if (get_current_workarea(&xpos, &ypos, &width, &height) < 0)
324          attribs.backing_store = Always;                  {
325                            error("Failed to get workarea.\n");
326                            error("Perhaps your window manager does not support EWMH?\n");
327                            error("Defaulting to geometry 800x600\n");
328                            width = 800;
329                            height = 600;
330                    }
331            }
332    
333          if (fullscreen)          if (fullscreen)
334          {          {
335                  attribs.override_redirect = True;                  width = WidthOfScreen(screen);
336                  width = WidthOfScreen(DefaultScreenOfDisplay(display));                  height = HeightOfScreen(screen);
                 height = HeightOfScreen(DefaultScreenOfDisplay(display));  
                 XSetInputFocus(display, PointerRoot, RevertToPointerRoot,  
                                CurrentTime);  
337          }          }
338          else  
339            /* make sure width is a multiple of 4 */
340            width = (width + 3) & ~3;
341    
342            if (ownbackstore)
343          {          {
344                  attribs.override_redirect = False;                  backstore =
345                            XCreatePixmap(display, RootWindowOfScreen(screen), width, height, depth);
346    
347                    /* clear to prevent rubbish being exposed at startup */
348                    XSetForeground(display, gc, BlackPixelOfScreen(screen));
349                    XFillRectangle(display, backstore, gc, 0, 0, width, height);
350          }          }
351    
352          wnd = XCreateWindow(display, DefaultRootWindow(display),          mod_map = XGetModifierMapping(display);
353                              0, 0, width, height, 0, CopyFromParent,  
354                              InputOutput, CopyFromParent,          if (enable_compose)
355                              CWBackingStore | CWBackPixel | CWOverrideRedirect,                  IM = XOpenIM(display, NULL, NULL, NULL);
356                              &attribs);  
357            xkeymap_init();
358            return True;
359    }
360    
361    void
362    ui_deinit(void)
363    {
364            if (IM != NULL)
365                    XCloseIM(IM);
366    
367            XFreeModifiermap(mod_map);
368    
369            if (ownbackstore)
370                    XFreePixmap(display, backstore);
371    
372            XFreeGC(display, gc);
373            XCloseDisplay(display);
374            display = NULL;
375    }
376    
377    BOOL
378    ui_create_window(void)
379    {
380            XSetWindowAttributes attribs;
381            XClassHint *classhints;
382            XSizeHints *sizehints;
383            int wndwidth, wndheight;
384            long input_mask, ic_input_mask;
385            XEvent xevent;
386    
387            wndwidth = fullscreen ? WidthOfScreen(screen) : width;
388            wndheight = fullscreen ? HeightOfScreen(screen) : height;
389    
390            attribs.background_pixel = BlackPixelOfScreen(screen);
391            attribs.backing_store = ownbackstore ? NotUseful : Always;
392            attribs.override_redirect = fullscreen;
393    
394            wnd = XCreateWindow(display, RootWindowOfScreen(screen), 0, 0, wndwidth, wndheight,
395                                0, CopyFromParent, InputOutput, CopyFromParent,
396                                CWBackPixel | CWBackingStore | CWOverrideRedirect, &attribs);
397    
398          XStoreName(display, wnd, title);          XStoreName(display, wnd, title);
399    
400            if (hide_decorations)
401                    mwm_hide_decorations();
402    
403          classhints = XAllocClassHint();          classhints = XAllocClassHint();
404          if (classhints != NULL)          if (classhints != NULL)
   
405          {          {
406                  classhints->res_name = "rdesktop";                  classhints->res_name = classhints->res_class = "rdesktop";
                 classhints->res_class = "rdesktop";  
407                  XSetClassHint(display, wnd, classhints);                  XSetClassHint(display, wnd, classhints);
408                  XFree(classhints);                  XFree(classhints);
409          }          }
# Line 168  ui_create_window(char *title) Line 411  ui_create_window(char *title)
411          sizehints = XAllocSizeHints();          sizehints = XAllocSizeHints();
412          if (sizehints)          if (sizehints)
413          {          {
414                  sizehints->flags =                  sizehints->flags = PMinSize | PMaxSize;
415                          PPosition | PSize | PMinSize | PMaxSize | PBaseSize;                  sizehints->min_width = sizehints->max_width = width;
416                  sizehints->min_width = width;                  sizehints->min_height = sizehints->max_height = height;
                 sizehints->max_width = width;  
                 sizehints->min_height = height;  
                 sizehints->max_height = height;  
                 sizehints->base_width = width;  
                 sizehints->base_height = height;  
417                  XSetWMNormalHints(display, wnd, sizehints);                  XSetWMNormalHints(display, wnd, sizehints);
418                  XFree(sizehints);                  XFree(sizehints);
419          }          }
420    
421          input_mask = KeyPressMask | KeyReleaseMask;          input_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask |
422          input_mask |= ButtonPressMask | ButtonReleaseMask;                  VisibilityChangeMask | FocusChangeMask;
423          if (motion)  
424            if (sendmotion)
425                  input_mask |= PointerMotionMask;                  input_mask |= PointerMotionMask;
426            if (ownbackstore)
427                    input_mask |= ExposureMask;
428            if (fullscreen || grab_keyboard)
429                    input_mask |= EnterWindowMask;
430          if (grab_keyboard)          if (grab_keyboard)
431                  input_mask |= EnterWindowMask | LeaveWindowMask;                  input_mask |= LeaveWindowMask;
432    
433          XSelectInput(display, wnd, input_mask);          if (IM != NULL)
434          gc = XCreateGC(display, wnd, 0, NULL);          {
435                    IC = XCreateIC(IM, XNInputStyle, (XIMPreeditNothing | XIMStatusNothing),
436                                   XNClientWindow, wnd, XNFocusWindow, wnd, NULL);
437    
438                    if ((IC != NULL)
439                        && (XGetICValues(IC, XNFilterEvents, &ic_input_mask, NULL) == NULL))
440                            input_mask |= ic_input_mask;
441            }
442    
443            XSelectInput(display, wnd, input_mask);
444          XMapWindow(display, wnd);          XMapWindow(display, wnd);
445    
446            /* wait for VisibilityNotify */
447            do
448            {
449                    XMaskEvent(display, VisibilityChangeMask, &xevent);
450            }
451            while (xevent.type != VisibilityNotify);
452    
453            focused = False;
454            mouse_in_wnd = False;
455    
456            /* handle the WM_DELETE_WINDOW protocol */
457            protocol_atom = XInternAtom(display, "WM_PROTOCOLS", True);
458            kill_atom = XInternAtom(display, "WM_DELETE_WINDOW", True);
459            XSetWMProtocols(display, wnd, &kill_atom, 1);
460    
461          return True;          return True;
462  }  }
463    
464  void  void
465  ui_destroy_window()  ui_destroy_window(void)
466  {  {
467          XFreeGC(display, gc);          if (IC != NULL)
468                    XDestroyIC(IC);
469    
470          XDestroyWindow(display, wnd);          XDestroyWindow(display, wnd);
         XCloseDisplay(display);  
         display = NULL;  
471  }  }
472    
473  static uint8  void
474  xwin_translate_key(unsigned long key)  xwin_toggle_fullscreen(void)
475  {  {
476          DEBUG("KEY(code=0x%lx)\n", key);          Pixmap contents = 0;
477    
478            if (!ownbackstore)
479            {
480                    /* need to save contents of window */
481                    contents = XCreatePixmap(display, wnd, width, height, depth);
482                    XCopyArea(display, wnd, contents, gc, 0, 0, width, height, 0, 0);
483            }
484    
485          if ((key > 8) && (key <= 0x60))          ui_destroy_window();
486                  return (key - 8);          fullscreen = !fullscreen;
487            ui_create_window();
488    
489          switch (key)          XDefineCursor(display, wnd, current_cursor);
         {  
                 case 0x61:      /* home */  
                         return 0x47 | 0x80;  
                 case 0x62:      /* up arrow */  
                         return 0x48 | 0x80;  
                 case 0x63:      /* page up */  
                         return 0x49 | 0x80;  
                 case 0x64:      /* left arrow */  
                         return 0x4b | 0x80;  
                 case 0x66:      /* right arrow */  
                         return 0x4d | 0x80;  
                 case 0x67:      /* end */  
                         return 0x4f | 0x80;  
                 case 0x68:      /* down arrow */  
                         return 0x50 | 0x80;  
                 case 0x69:      /* page down */  
                         return 0x51 | 0x80;  
                 case 0x6a:      /* insert */  
                         return 0x52 | 0x80;  
                 case 0x6b:      /* delete */  
                         return 0x53 | 0x80;  
                 case 0x6c:      /* keypad enter */  
                         return 0x1c | 0x80;  
                 case 0x6d:      /* right ctrl */  
                         return 0x1d | 0x80;  
                 case 0x6f:      /* ctrl - print screen */  
                         return 0x37 | 0x80;  
                 case 0x70:      /* keypad '/' */  
                         return 0x35 | 0x80;  
                 case 0x71:      /* right alt */  
                         return 0x38 | 0x80;  
                 case 0x72:      /* ctrl break */  
                         return 0x46 | 0x80;  
                 case 0x73:      /* left window key */  
                         return 0xff;    /* real scancode is 5b */  
                 case 0x74:      /* right window key */  
                         return 0xff;    /* real scancode is 5c */  
                 case 0x75:      /* menu key */  
                         return 0x5d | 0x80;  
         }  
   
         return 0;  
 }  
   
 static uint16  
 xwin_translate_mouse(unsigned long button)  
 {  
         switch (button)  
         {  
                 case Button1:   /* left */  
                         return MOUSE_FLAG_BUTTON1;  
                 case Button2:   /* middle */  
                         return MOUSE_FLAG_BUTTON3;  
                 case Button3:   /* right */  
                         return MOUSE_FLAG_BUTTON2;  
         }  
   
         return 0;  
 }  
   
 void  
 ui_process_events()  
 {  
         XEvent event;  
         uint8 scancode;  
         uint16 button;  
         uint32 ev_time;  
490    
491          if (display == NULL)          if (!ownbackstore)
492                  return;          {
493                    XCopyArea(display, contents, wnd, gc, 0, 0, width, height, 0, 0);
494                    XFreePixmap(display, contents);
495            }
496    }
497    
498    /* Process all events in Xlib queue
499       Returns 0 after user quit, 1 otherwise */
500    static int
501    xwin_process_events(void)
502    {
503            XEvent xevent;
504            KeySym keysym;
505            uint16 button, flags;
506            uint32 ev_time;
507            key_translation tr;
508            char str[256];
509            Status status;
510            unsigned int state;
511            Window wdummy;
512            int dummy;
513    
514          while (XCheckWindowEvent(display, wnd, ~0, &event))          while (XPending(display) > 0)
515          {          {
516                  ev_time = time(NULL);                  XNextEvent(display, &xevent);
517    
518                    if ((IC != NULL) && (XFilterEvent(&xevent, None) == True))
519                    {
520                            DEBUG_KBD(("Filtering event\n"));
521                            continue;
522                    }
523    
524                  switch (event.type)                  flags = 0;
525    
526                    switch (xevent.type)
527                  {                  {
528                            case ClientMessage:
529                                    /* the window manager told us to quit */
530                                    if ((xevent.xclient.message_type == protocol_atom)
531                                        && (xevent.xclient.data.l[0] == kill_atom))
532                                            /* Quit */
533                                            return 0;
534                                    break;
535    
536                          case KeyPress:                          case KeyPress:
537                                  scancode = xwin_translate_key(event.xkey.keycode);                                  if (IC != NULL)
538                                  if (scancode == 0)                                          /* Multi_key compatible version */
539                                    {
540                                            XmbLookupString(IC,
541                                                            (XKeyPressedEvent *) &
542                                                            xevent, str, sizeof(str), &keysym, &status);
543                                            if (!((status == XLookupKeySym) || (status == XLookupBoth)))
544                                            {
545                                                    error("XmbLookupString failed with status 0x%x\n",
546                                                          status);
547                                                    break;
548                                            }
549                                    }
550                                    else
551                                    {
552                                            /* Plain old XLookupString */
553                                            DEBUG_KBD(("\nNo input context, using XLookupString\n"));
554                                            XLookupString((XKeyEvent *) & xevent,
555                                                          str, sizeof(str), &keysym, NULL);
556                                    }
557    
558                                    DEBUG_KBD(("KeyPress for (keysym 0x%lx, %s)\n", keysym,
559                                               get_ksname(keysym)));
560    
561                                    ev_time = time(NULL);
562                                    if (handle_special_keys(keysym, xevent.xkey.state, ev_time, True))
563                                            break;
564    
565                                    tr = xkeymap_translate_key(keysym,
566                                                               xevent.xkey.keycode, xevent.xkey.state);
567    
568                                    if (tr.scancode == 0)
569                                          break;                                          break;
570    
571                                  rdp_send_input(ev_time, RDP_INPUT_SCANCODE, 0,                                  ensure_remote_modifiers(ev_time, tr);
572                                                 scancode, 0);  
573                                    rdp_send_scancode(ev_time, RDP_KEYPRESS, tr.scancode);
574                                  break;                                  break;
575    
576                          case KeyRelease:                          case KeyRelease:
577                                  scancode = xwin_translate_key(event.xkey.keycode);                                  XLookupString((XKeyEvent *) & xevent, str,
578                                  if (scancode == 0)                                                sizeof(str), &keysym, NULL);
579    
580                                    DEBUG_KBD(("\nKeyRelease for (keysym 0x%lx, %s)\n", keysym,
581                                               get_ksname(keysym)));
582    
583                                    ev_time = time(NULL);
584                                    if (handle_special_keys(keysym, xevent.xkey.state, ev_time, False))
585                                          break;                                          break;
586    
587                                  rdp_send_input(ev_time, RDP_INPUT_SCANCODE,                                  tr = xkeymap_translate_key(keysym,
588                                                 KBD_FLAG_DOWN | KBD_FLAG_UP,                                                             xevent.xkey.keycode, xevent.xkey.state);
                                                scancode, 0);  
                                 break;  
589    
590                          case ButtonPress:                                  if (tr.scancode == 0)
                                 button = xwin_translate_mouse(event.xbutton.button);  
                                 if (button == 0)  
591                                          break;                                          break;
592    
593                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  rdp_send_scancode(ev_time, RDP_KEYRELEASE, tr.scancode);
                                                button | MOUSE_FLAG_DOWN,  
                                                event.xbutton.x,  
                                                event.xbutton.y);  
594                                  break;                                  break;
595    
596                            case ButtonPress:
597                                    flags = MOUSE_FLAG_DOWN;
598                                    /* fall through */
599    
600                          case ButtonRelease:                          case ButtonRelease:
601                                  button = xwin_translate_mouse(event.xbutton.button);                                  button = xkeymap_translate_button(xevent.xbutton.button);
602                                  if (button == 0)                                  if (button == 0)
603                                          break;                                          break;
604    
605                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  rdp_send_input(time(NULL), RDP_INPUT_MOUSE,
606                                                 button,                                                 flags | button, xevent.xbutton.x, xevent.xbutton.y);
                                                event.xbutton.x,  
                                                event.xbutton.y);  
607                                  break;                                  break;
608    
609                          case MotionNotify:                          case MotionNotify:
610                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  if (fullscreen && !focused)
611                                                 MOUSE_FLAG_MOVE,                                          XSetInputFocus(display, wnd, RevertToPointerRoot,
612                                                 event.xmotion.x,                                                         CurrentTime);
613                                                 event.xmotion.y);                                  rdp_send_input(time(NULL), RDP_INPUT_MOUSE,
614                                                   MOUSE_FLAG_MOVE, xevent.xmotion.x, xevent.xmotion.y);
615                                    break;
616    
617                            case FocusIn:
618                                    if (xevent.xfocus.mode == NotifyGrab)
619                                            break;
620                                    focused = True;
621                                    XQueryPointer(display, wnd, &wdummy, &wdummy, &dummy, &dummy,
622                                                  &dummy, &dummy, &state);
623                                    reset_modifier_keys(state);
624                                    if (grab_keyboard && mouse_in_wnd)
625                                            XGrabKeyboard(display, wnd, True,
626                                                          GrabModeAsync, GrabModeAsync, CurrentTime);
627                                    break;
628    
629                            case FocusOut:
630                                    if (xevent.xfocus.mode == NotifyUngrab)
631                                            break;
632                                    focused = False;
633                                    if (xevent.xfocus.mode == NotifyWhileGrabbed)
634                                            XUngrabKeyboard(display, CurrentTime);
635                                  break;                                  break;
636    
637                          case EnterNotify:                          case EnterNotify:
638                                  if (grab_keyboard)                                  /* we only register for this event when in fullscreen mode */
639                                          xwin_grab_keyboard();                                  /* or grab_keyboard */
640                                    mouse_in_wnd = True;
641                                    if (fullscreen)
642                                    {
643                                            XSetInputFocus(display, wnd, RevertToPointerRoot,
644                                                           CurrentTime);
645                                            break;
646                                    }
647                                    if (focused)
648                                            XGrabKeyboard(display, wnd, True,
649                                                          GrabModeAsync, GrabModeAsync, CurrentTime);
650                                  break;                                  break;
651    
652                          case LeaveNotify:                          case LeaveNotify:
653                                  if (grab_keyboard)                                  /* we only register for this event when grab_keyboard */
654                                          xwin_ungrab_keyboard();                                  mouse_in_wnd = False;
655                                    XUngrabKeyboard(display, CurrentTime);
656                                    break;
657    
658                            case Expose:
659                                    XCopyArea(display, backstore, wnd, gc,
660                                              xevent.xexpose.x, xevent.xexpose.y,
661                                              xevent.xexpose.width,
662                                              xevent.xexpose.height,
663                                              xevent.xexpose.x, xevent.xexpose.y);
664                                    break;
665    
666                            case MappingNotify:
667                                    /* Refresh keyboard mapping if it has changed. This is important for
668                                       Xvnc, since it allocates keycodes dynamically */
669                                    if (xevent.xmapping.request == MappingKeyboard
670                                        || xevent.xmapping.request == MappingModifier)
671                                            XRefreshKeyboardMapping(&xevent.xmapping);
672    
673                                    if (xevent.xmapping.request == MappingModifier)
674                                    {
675                                            XFreeModifiermap(mod_map);
676                                            mod_map = XGetModifierMapping(display);
677                                    }
678                                  break;                                  break;
679    
680                  }                  }
681          }          }
682            /* Keep going */
683            return 1;
684    }
685    
686    /* Returns 0 after user quit, 1 otherwise */
687    int
688    ui_select(int rdp_socket)
689    {
690            int n = (rdp_socket > x_socket) ? rdp_socket + 1 : x_socket + 1;
691            fd_set rfds;
692    
693            FD_ZERO(&rfds);
694    
695            while (True)
696            {
697                    /* Process any events already waiting */
698                    if (!xwin_process_events())
699                            /* User quit */
700                            return 0;
701    
702                    FD_ZERO(&rfds);
703                    FD_SET(rdp_socket, &rfds);
704                    FD_SET(x_socket, &rfds);
705    
706                    switch (select(n, &rfds, NULL, NULL, NULL))
707                    {
708                            case -1:
709                                    error("select: %s\n", strerror(errno));
710    
711                            case 0:
712                                    continue;
713                    }
714    
715                    if (FD_ISSET(rdp_socket, &rfds))
716                            return 1;
717            }
718  }  }
719    
720  void  void
# Line 357  ui_move_pointer(int x, int y) Line 724  ui_move_pointer(int x, int y)
724  }  }
725    
726  HBITMAP  HBITMAP
727  ui_create_bitmap(int width, int height, uint8 *data)  ui_create_bitmap(int width, int height, uint8 * data)
728  {  {
729          XImage *image;          XImage *image;
730          Pixmap bitmap;          Pixmap bitmap;
731          uint8 *tdata;          uint8 *tdata;
732          tdata = (private_colormap ? data : translate(width, height, data));  
733            tdata = (owncolmap ? data : translate_image(width, height, data));
734          bitmap = XCreatePixmap(display, wnd, width, height, depth);          bitmap = XCreatePixmap(display, wnd, width, height, depth);
735          image =          image = XCreateImage(display, visual, depth, ZPixmap, 0,
736                  XCreateImage(display, visual,                               (char *) tdata, width, height, 8, 0);
                              depth, ZPixmap,  
                              0, tdata, width, height, BitmapPad(display), 0);  
737    
         xwin_set_function(ROP2_COPY);  
738          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);
739    
740          XFree(image);          XFree(image);
741          if (!private_colormap)          if (!owncolmap)
742                  xfree(tdata);                  xfree(tdata);
743          return (HBITMAP) bitmap;          return (HBITMAP) bitmap;
744  }  }
745    
746  void  void
747  ui_paint_bitmap(int x, int y, int cx, int cy,  ui_paint_bitmap(int x, int y, int cx, int cy, int width, int height, uint8 * data)
                 int width, int height, uint8 *data)  
748  {  {
749          XImage *image;          XImage *image;
750          uint8 *tdata =          uint8 *tdata;
                 (private_colormap ? data : translate(width, height, data));  
         image =  
                 XCreateImage(display, visual, depth, ZPixmap, 0, tdata, width,  
                              height, BitmapPad(display), 0);  
   
         xwin_set_function(ROP2_COPY);  
   
         /* Window */  
         XPutImage(display, wnd, gc, image, 0, 0, x, y, cx, cy);  
         XFree(image);  
         if (!private_colormap)  
                 xfree(tdata);  
 }  
   
 void  
 ui_destroy_bitmap(HBITMAP bmp)  
 {  
         XFreePixmap(display, (Pixmap) bmp);  
 }  
751    
752  HCURSOR          tdata = (owncolmap ? data : translate_image(width, height, data));
753  ui_create_cursor(unsigned int x, unsigned int y, int width,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
754                   int height, uint8 *mask, uint8 *data)                               (char *) tdata, width, height, 8, 0);
 {  
         XImage *imagecursor;  
         XImage *imagemask;  
         Pixmap maskbitmap, cursorbitmap;  
         Cursor cursor;  
         XColor bg, fg;  
         GC lgc;  
         int i, x1, y1, scanlinelen;  
         uint8 *cdata, *cmask;  
         uint8 c;  
         cdata = (uint8 *) malloc(sizeof(uint8) * width * height);  
         if (!cdata)  
                 return NULL;  
         scanlinelen = (width + 7) >> 3;  
         cmask = (uint8 *) malloc(sizeof(uint8) * scanlinelen * height);  
         if (!cmask)  
         {  
                 free(cdata);  
                 return NULL;  
         }  
         i = (height - 1) * scanlinelen;  
755    
756          if (!screen_msbfirst)          if (ownbackstore)
757          {          {
758                  while (i >= 0)                  XPutImage(display, backstore, gc, image, 0, 0, x, y, cx, cy);
759                  {                  XCopyArea(display, backstore, wnd, gc, x, y, cx, cy, x, y);
                         for (x1 = 0; x1 < scanlinelen; x1++)  
                         {  
                                 c = *(mask++);  
                                 cmask[i + x1] =  
                                         ((c & 0x1) << 7) | ((c & 0x2) << 5) |  
                                         ((c & 0x4) << 3) | ((c & 0x8) << 1) |  
                                         ((c & 0x10) >> 1) | ((c & 0x20) >> 3)  
                                         | ((c & 0x40) >> 5) | ((c & 0x80) >>  
                                                                7);  
                         }  
                         i -= scanlinelen;  
                 }  
760          }          }
761          else          else
762          {          {
763                  while (i >= 0)                  XPutImage(display, wnd, gc, image, 0, 0, x, y, cx, cy);
                 {  
                         for (x1 = 0; x1 < scanlinelen; x1++)  
                         {  
                                 cmask[i + x1] = *(mask++);  
                         }  
                         i -= scanlinelen;  
                 }  
764          }          }
765    
766            XFree(image);
767          fg.red = 0;          if (!owncolmap)
768          fg.blue = 0;                  xfree(tdata);
         fg.green = 0;  
         fg.flags = DoRed | DoBlue | DoGreen;  
         bg.red = 65535;  
         bg.blue = 65535;  
         bg.green = 65535;  
         bg.flags = DoRed | DoBlue | DoGreen;  
         maskbitmap = XCreatePixmap(display, wnd, width, height, 1);  
         cursorbitmap = XCreatePixmap(display, wnd, width, height, 1);  
         lgc = XCreateGC(display, maskbitmap, 0, NULL);  
         XSetFunction(display, lgc, GXcopy);  
         imagemask =  
                 XCreateImage(display, visual, 1, XYBitmap, 0, cmask, width,  
                              height, 8, 0);  
         imagecursor =  
                 XCreateImage(display, visual, 1, XYBitmap, 0, cdata, width,  
                              height, 8, 0);  
         for (y1 = height - 1; y1 >= 0; y1--)  
                 for (x1 = 0; x1 < width; x1++)  
                 {  
                         if (data[0] >= 0x80 || data[1] >= 0x80  
                             || data[2] >= 0x80)  
                                 if (XGetPixel(imagemask, x1, y1))  
   
                                 {  
                                         XPutPixel(imagecursor, x1, y1, 0);  
                                         XPutPixel(imagemask, x1, y1, 0);        /* mask is blank for text cursor! */  
                                 }  
   
                                 else  
                                         XPutPixel(imagecursor, x1, y1, 1);  
   
                         else  
                                 XPutPixel(imagecursor, x1, y1,  
                                           XGetPixel(imagemask, x1, y1));  
                         data += 3;  
                 }  
         XPutImage(display, maskbitmap, lgc, imagemask, 0, 0, 0, 0, width,  
                   height);  
         XPutImage(display, cursorbitmap, lgc, imagecursor, 0, 0, 0, 0, width,  
                   height); XFree(imagemask);  
         XFree(imagecursor);  
         free(cmask);  
         free(cdata);  
         XFreeGC(display, lgc);  
         cursor =  
                 XCreatePixmapCursor(display, cursorbitmap, maskbitmap, &fg,  
                                     &bg, x, y);  
         XFreePixmap(display, maskbitmap);  
         XFreePixmap(display, cursorbitmap);  
         return (HCURSOR) cursor;  
 }  
   
 void  
 ui_set_cursor(HCURSOR cursor)  
 {  
         XDefineCursor(display, wnd, (Cursor) cursor);  
769  }  }
770    
771  void  void
772  ui_destroy_cursor(HCURSOR cursor)  ui_destroy_bitmap(HBITMAP bmp)
773  {  {
774          XFreeCursor(display, (Cursor) cursor);          XFreePixmap(display, (Pixmap) bmp);
775  }  }
776    
777  HGLYPH  HGLYPH
778  ui_create_glyph(int width, int height, uint8 *data)  ui_create_glyph(int width, int height, uint8 * data)
779  {  {
780          XImage *image;          XImage *image;
781          Pixmap bitmap;          Pixmap bitmap;
# Line 538  ui_create_glyph(int width, int height, u Line 787  ui_create_glyph(int width, int height, u
787          bitmap = XCreatePixmap(display, wnd, width, height, 1);          bitmap = XCreatePixmap(display, wnd, width, height, 1);
788          gc = XCreateGC(display, bitmap, 0, NULL);          gc = XCreateGC(display, bitmap, 0, NULL);
789    
790          image = XCreateImage(display, visual, 1, ZPixmap, 0,          image = XCreateImage(display, visual, 1, ZPixmap, 0, (char *) data,
791                               data, width, height, 8, scanline);                               width, height, 8, scanline);
792          image->byte_order = MSBFirst;          image->byte_order = MSBFirst;
793          image->bitmap_bit_order = MSBFirst;          image->bitmap_bit_order = MSBFirst;
794          XInitImage(image);          XInitImage(image);
795    
         XSetFunction(display, gc, GXcopy);  
796          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);
797    
798          XFree(image);          XFree(image);
799          XFreeGC(display, gc);          XFreeGC(display, gc);
   
800          return (HGLYPH) bitmap;          return (HGLYPH) bitmap;
801  }  }
802    
# Line 558  ui_destroy_glyph(HGLYPH glyph) Line 806  ui_destroy_glyph(HGLYPH glyph)
806          XFreePixmap(display, (Pixmap) glyph);          XFreePixmap(display, (Pixmap) glyph);
807  }  }
808    
809  HCOLOURMAP  HCURSOR
810  ui_create_colourmap(COLOURMAP *colours)  ui_create_cursor(unsigned int x, unsigned int y, int width, int height,
811                     uint8 * andmask, uint8 * xormask)
812  {  {
813          if (!private_colormap)          HGLYPH maskglyph, cursorglyph;
814            XColor bg, fg;
815            Cursor xcursor;
816            uint8 *cursor, *pcursor;
817            uint8 *mask, *pmask;
818            uint8 nextbit;
819            int scanline, offset;
820            int i, j;
821    
822            scanline = (width + 7) / 8;
823            offset = scanline * height;
824    
825            cursor = xmalloc(offset);
826            memset(cursor, 0, offset);
827    
828            mask = xmalloc(offset);
829            memset(mask, 0, offset);
830    
831            /* approximate AND and XOR masks with a monochrome X pointer */
832            for (i = 0; i < height; i++)
833          {          {
834                  COLOURENTRY *entry;                  offset -= scanline;
835                  int i, ncolours = colours->ncolours;                  pcursor = &cursor[offset];
836                  uint32 *nc = xmalloc(sizeof(*colmap) * ncolours);                  pmask = &mask[offset];
837    
838                    for (j = 0; j < scanline; j++)
839                    {
840                            for (nextbit = 0x80; nextbit != 0; nextbit >>= 1)
841                            {
842                                    if (xormask[0] || xormask[1] || xormask[2])
843                                    {
844                                            *pcursor |= (~(*andmask) & nextbit);
845                                            *pmask |= nextbit;
846                                    }
847                                    else
848                                    {
849                                            *pcursor |= ((*andmask) & nextbit);
850                                            *pmask |= (~(*andmask) & nextbit);
851                                    }
852    
853                                    xormask += 3;
854                            }
855    
856                            andmask++;
857                            pcursor++;
858                            pmask++;
859                    }
860            }
861    
862            fg.red = fg.blue = fg.green = 0xffff;
863            bg.red = bg.blue = bg.green = 0x0000;
864            fg.flags = bg.flags = DoRed | DoBlue | DoGreen;
865    
866            cursorglyph = ui_create_glyph(width, height, cursor);
867            maskglyph = ui_create_glyph(width, height, mask);
868    
869            xcursor =
870                    XCreatePixmapCursor(display, (Pixmap) cursorglyph,
871                                        (Pixmap) maskglyph, &fg, &bg, x, y);
872    
873            ui_destroy_glyph(maskglyph);
874            ui_destroy_glyph(cursorglyph);
875            xfree(mask);
876            xfree(cursor);
877            return (HCURSOR) xcursor;
878    }
879    
880    void
881    ui_set_cursor(HCURSOR cursor)
882    {
883            current_cursor = (Cursor) cursor;
884            XDefineCursor(display, wnd, current_cursor);
885    }
886    
887    void
888    ui_destroy_cursor(HCURSOR cursor)
889    {
890            XFreeCursor(display, (Cursor) cursor);
891    }
892    
893    #define MAKE_XCOLOR(xc,c) \
894                    (xc)->red   = ((c)->red   << 8) | (c)->red; \
895                    (xc)->green = ((c)->green << 8) | (c)->green; \
896                    (xc)->blue  = ((c)->blue  << 8) | (c)->blue; \
897                    (xc)->flags = DoRed | DoGreen | DoBlue;
898    
899    
900    HCOLOURMAP
901    ui_create_colourmap(COLOURMAP * colours)
902    {
903            COLOURENTRY *entry;
904            int i, ncolours = colours->ncolours;
905            if (!owncolmap)
906            {
907                    uint32 *map = xmalloc(sizeof(*colmap) * ncolours);
908                    XColor xentry;
909                    XColor xc_cache[256];
910                    uint32 colour;
911                    int colLookup = 256;
912                  for (i = 0; i < ncolours; i++)                  for (i = 0; i < ncolours; i++)
913                  {                  {
                         XColor xc;  
914                          entry = &colours->colours[i];                          entry = &colours->colours[i];
915                          xc.red = entry->red << 8;                          MAKE_XCOLOR(&xentry, entry);
916                          xc.green = entry->green << 8;  
917                          xc.blue = entry->blue << 8;                          if (XAllocColor(display, xcolmap, &xentry) == 0)
918                          XAllocColor(display,                          {
919                                      DefaultColormap(display,                                  /* Allocation failed, find closest match. */
920                                                      DefaultScreen(display)),                                  int j = 256;
921                                      &xc);                                  int nMinDist = 3 * 256 * 256;
922                          /* XXX Check return value */                                  long nDist = nMinDist;
923                          nc[i] = xc.pixel;  
924                                    /* only get the colors once */
925                                    while (colLookup--)
926                                    {
927                                            xc_cache[colLookup].pixel = colLookup;
928                                            xc_cache[colLookup].red = xc_cache[colLookup].green =
929                                                    xc_cache[colLookup].blue = 0;
930                                            xc_cache[colLookup].flags = 0;
931                                            XQueryColor(display,
932                                                        DefaultColormap(display,
933                                                                        DefaultScreen(display)),
934                                                        &xc_cache[colLookup]);
935                                    }
936                                    colLookup = 0;
937    
938                                    /* approximate the pixel */
939                                    while (j--)
940                                    {
941                                            if (xc_cache[j].flags)
942                                            {
943                                                    nDist = ((long) (xc_cache[j].red >> 8) -
944                                                             (long) (xentry.red >> 8)) *
945                                                            ((long) (xc_cache[j].red >> 8) -
946                                                             (long) (xentry.red >> 8)) +
947                                                            ((long) (xc_cache[j].green >> 8) -
948                                                             (long) (xentry.green >> 8)) *
949                                                            ((long) (xc_cache[j].green >> 8) -
950                                                             (long) (xentry.green >> 8)) +
951                                                            ((long) (xc_cache[j].blue >> 8) -
952                                                             (long) (xentry.blue >> 8)) *
953                                                            ((long) (xc_cache[j].blue >> 8) -
954                                                             (long) (xentry.blue >> 8));
955                                            }
956                                            if (nDist < nMinDist)
957                                            {
958                                                    nMinDist = nDist;
959                                                    xentry.pixel = j;
960                                            }
961                                    }
962                            }
963                            colour = xentry.pixel;
964    
965                            /* update our cache */
966                            if (xentry.pixel < 256)
967                            {
968                                    xc_cache[xentry.pixel].red = xentry.red;
969                                    xc_cache[xentry.pixel].green = xentry.green;
970                                    xc_cache[xentry.pixel].blue = xentry.blue;
971    
972                            }
973    
974    
975                            /* byte swap here to make translate_image faster */
976                            map[i] = translate_colour(colour);
977                  }                  }
978                  return nc;                  return map;
979          }          }
980          else          else
981          {          {
                 COLOURENTRY *entry;  
982                  XColor *xcolours, *xentry;                  XColor *xcolours, *xentry;
983                  Colormap map;                  Colormap map;
984                  int i, ncolours = colours->ncolours;  
985                  xcolours = xmalloc(sizeof(XColor) * ncolours);                  xcolours = xmalloc(sizeof(XColor) * ncolours);
986                  for (i = 0; i < ncolours; i++)                  for (i = 0; i < ncolours; i++)
987                  {                  {
988                          entry = &colours->colours[i];                          entry = &colours->colours[i];
989                          xentry = &xcolours[i];                          xentry = &xcolours[i];
   
990                          xentry->pixel = i;                          xentry->pixel = i;
991                          xentry->red = entry->red << 8;                          MAKE_XCOLOR(xentry, entry);
                         xentry->blue = entry->blue << 8;  
                         xentry->green = entry->green << 8;  
                         xentry->flags = DoRed | DoBlue | DoGreen;  
992                  }                  }
993    
994                  map = XCreateColormap(display, wnd, visual, AllocAll);                  map = XCreateColormap(display, wnd, visual, AllocAll);
# Line 612  ui_create_colourmap(COLOURMAP *colours) Line 1002  ui_create_colourmap(COLOURMAP *colours)
1002  void  void
1003  ui_destroy_colourmap(HCOLOURMAP map)  ui_destroy_colourmap(HCOLOURMAP map)
1004  {  {
1005          XFreeColormap(display, (Colormap) map);          if (!owncolmap)
1006                    xfree(map);
1007            else
1008                    XFreeColormap(display, (Colormap) map);
1009  }  }
1010    
1011  void  void
1012  ui_set_colourmap(HCOLOURMAP map)  ui_set_colourmap(HCOLOURMAP map)
1013  {  {
1014            if (!owncolmap)
         /* XXX, change values of all pixels on the screen if the new colmap  
          * doesn't have the same values as the old one? */  
         if (!private_colormap)  
1015                  colmap = map;                  colmap = map;
1016          else          else
         {  
1017                  XSetWindowColormap(display, wnd, (Colormap) map);                  XSetWindowColormap(display, wnd, (Colormap) map);
                 if (fullscreen)  
                         XInstallColormap(display, (Colormap) map);  
         }  
1018  }  }
1019    
1020  void  void
# Line 644  ui_set_clip(int x, int y, int cx, int cy Line 1030  ui_set_clip(int x, int y, int cx, int cy
1030  }  }
1031    
1032  void  void
1033  ui_reset_clip()  ui_reset_clip(void)
1034  {  {
1035          XRectangle rect;          XRectangle rect;
1036    
# Line 656  ui_reset_clip() Line 1042  ui_reset_clip()
1042  }  }
1043    
1044  void  void
1045  ui_bell()  ui_bell(void)
1046  {  {
1047          XBell(display, 0);          XBell(display, 0);
1048  }  }
# Line 665  void Line 1051  void
1051  ui_destblt(uint8 opcode,  ui_destblt(uint8 opcode,
1052             /* dest */ int x, int y, int cx, int cy)             /* dest */ int x, int y, int cx, int cy)
1053  {  {
1054          xwin_set_function(opcode);          SET_FUNCTION(opcode);
1055            FILL_RECTANGLE(x, y, cx, cy);
1056          XFillRectangle(display, wnd, gc, x, y, cx, cy);          RESET_FUNCTION(opcode);
1057  }  }
1058    
1059  void  void
1060  ui_patblt(uint8 opcode,  ui_patblt(uint8 opcode,
1061            /* dest */ int x, int y, int cx, int cy,            /* dest */ int x, int y, int cx, int cy,
1062            /* brush */ BRUSH *brush, int bgcolour, int fgcolour)            /* brush */ BRUSH * brush, int bgcolour, int fgcolour)
1063  {  {
         Display *dpy = display;  
1064          Pixmap fill;          Pixmap fill;
1065          uint8 i, ipattern[8];          uint8 i, ipattern[8];
1066    
1067          xwin_set_function(opcode);          SET_FUNCTION(opcode);
1068    
1069          switch (brush->style)          switch (brush->style)
1070          {          {
1071                  case 0: /* Solid */                  case 0: /* Solid */
1072                          XSetForeground(dpy, gc, Ctrans(fgcolour));                          SET_FOREGROUND(fgcolour);
1073                          XFillRectangle(dpy, wnd, gc, x, y, cx, cy);                          FILL_RECTANGLE(x, y, cx, cy);
1074                          break;                          break;
1075    
1076                  case 3: /* Pattern */                  case 3: /* Pattern */
1077                          for (i = 0; i != 8; i++)                          for (i = 0; i != 8; i++)
1078                                  ipattern[i] = ~brush->pattern[i];                                  ipattern[7 - i] = brush->pattern[i];
1079                          fill = (Pixmap) ui_create_glyph(8, 8, ipattern);                          fill = (Pixmap) ui_create_glyph(8, 8, ipattern);
1080    
1081                          XSetForeground(dpy, gc, Ctrans(fgcolour));                          SET_FOREGROUND(bgcolour);
1082                          XSetBackground(dpy, gc, Ctrans(bgcolour));                          SET_BACKGROUND(fgcolour);
1083                          XSetFillStyle(dpy, gc, FillOpaqueStippled);                          XSetFillStyle(display, gc, FillOpaqueStippled);
1084                          XSetStipple(dpy, gc, fill);                          XSetStipple(display, gc, fill);
1085                            XSetTSOrigin(display, gc, brush->xorigin, brush->yorigin);
1086    
1087                          XFillRectangle(dpy, wnd, gc, x, y, cx, cy);                          FILL_RECTANGLE(x, y, cx, cy);
1088    
1089                          XSetFillStyle(dpy, gc, FillSolid);                          XSetFillStyle(display, gc, FillSolid);
1090                            XSetTSOrigin(display, gc, 0, 0);
1091                          ui_destroy_glyph((HGLYPH) fill);                          ui_destroy_glyph((HGLYPH) fill);
1092                          break;                          break;
1093    
1094                  default:                  default:
1095                          NOTIMP("brush %d\n", brush->style);                          unimpl("brush %d\n", brush->style);
1096          }          }
1097    
1098            RESET_FUNCTION(opcode);
1099  }  }
1100    
1101  void  void
# Line 714  ui_screenblt(uint8 opcode, Line 1103  ui_screenblt(uint8 opcode,
1103               /* dest */ int x, int y, int cx, int cy,               /* dest */ int x, int y, int cx, int cy,
1104               /* src */ int srcx, int srcy)               /* src */ int srcx, int srcy)
1105  {  {
1106          xwin_set_function(opcode);          SET_FUNCTION(opcode);
   
1107          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);
1108            if (ownbackstore)
1109                    XCopyArea(display, backstore, backstore, gc, srcx, srcy, cx, cy, x, y);
1110            RESET_FUNCTION(opcode);
1111  }  }
1112    
1113  void  void
# Line 724  ui_memblt(uint8 opcode, Line 1115  ui_memblt(uint8 opcode,
1115            /* dest */ int x, int y, int cx, int cy,            /* dest */ int x, int y, int cx, int cy,
1116            /* src */ HBITMAP src, int srcx, int srcy)            /* src */ HBITMAP src, int srcx, int srcy)
1117  {  {
1118          xwin_set_function(opcode);          SET_FUNCTION(opcode);
   
1119          XCopyArea(display, (Pixmap) src, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, (Pixmap) src, wnd, gc, srcx, srcy, cx, cy, x, y);
1120            if (ownbackstore)
1121                    XCopyArea(display, (Pixmap) src, backstore, gc, srcx, srcy, cx, cy, x, y);
1122            RESET_FUNCTION(opcode);
1123  }  }
1124    
1125  void  void
1126  ui_triblt(uint8 opcode,  ui_triblt(uint8 opcode,
1127            /* dest */ int x, int y, int cx, int cy,            /* dest */ int x, int y, int cx, int cy,
1128            /* src */ HBITMAP src, int srcx, int srcy,            /* src */ HBITMAP src, int srcx, int srcy,
1129            /* brush */ BRUSH *brush, int bgcolour, int fgcolour)            /* brush */ BRUSH * brush, int bgcolour, int fgcolour)
1130  {  {
1131          /* This is potentially difficult to do in general. Until someone          /* This is potentially difficult to do in general. Until someone
1132             comes up with a more efficient way of doing it I am using cases. */             comes up with a more efficient way of doing it I am using cases. */
# Line 742  ui_triblt(uint8 opcode, Line 1135  ui_triblt(uint8 opcode,
1135          {          {
1136                  case 0x69:      /* PDSxxn */                  case 0x69:      /* PDSxxn */
1137                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);
1138                          ui_patblt(ROP2_NXOR, x, y, cx, cy,                          ui_patblt(ROP2_NXOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1139                          break;                          break;
1140    
1141                  case 0xb8:      /* PSDPxax */                  case 0xb8:      /* PSDPxax */
1142                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1143                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);
1144                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1145                          break;                          break;
1146    
1147                  case 0xc0:                  case 0xc0:      /* PSa */
1148                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);
1149                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour,                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   fgcolour);  
1150                          break;                          break;
1151    
1152                  default:                  default:
1153                          NOTIMP("triblt 0x%x\n", opcode);                          unimpl("triblt 0x%x\n", opcode);
1154                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);
1155          }          }
1156  }  }
# Line 769  ui_triblt(uint8 opcode, Line 1158  ui_triblt(uint8 opcode,
1158  void  void
1159  ui_line(uint8 opcode,  ui_line(uint8 opcode,
1160          /* dest */ int startx, int starty, int endx, int endy,          /* dest */ int startx, int starty, int endx, int endy,
1161          /* pen */ PEN *pen)          /* pen */ PEN * pen)
1162  {  {
1163          xwin_set_function(opcode);          SET_FUNCTION(opcode);
1164            SET_FOREGROUND(pen->colour);
         XSetForeground(display, gc, Ctrans(pen->colour));  
1165          XDrawLine(display, wnd, gc, startx, starty, endx, endy);          XDrawLine(display, wnd, gc, startx, starty, endx, endy);
1166            if (ownbackstore)
1167                    XDrawLine(display, backstore, gc, startx, starty, endx, endy);
1168            RESET_FUNCTION(opcode);
1169  }  }
1170    
1171  void  void
# Line 782  ui_rect( Line 1173  ui_rect(
1173                 /* dest */ int x, int y, int cx, int cy,                 /* dest */ int x, int y, int cx, int cy,
1174                 /* brush */ int colour)                 /* brush */ int colour)
1175  {  {
1176          xwin_set_function(ROP2_COPY);          SET_FOREGROUND(colour);
1177            FILL_RECTANGLE(x, y, cx, cy);
         XSetForeground(display, gc, Ctrans(colour));  
         XFillRectangle(display, wnd, gc, x, y, cx, cy);  
1178  }  }
1179    
1180    /* warning, this function only draws on wnd or backstore, not both */
1181  void  void
1182  ui_draw_glyph(int mixmode,  ui_draw_glyph(int mixmode,
1183                /* dest */ int x, int y, int cx, int cy,                /* dest */ int x, int y, int cx, int cy,
1184                /* src */ HGLYPH glyph, int srcx, int srcy, int bgcolour,                /* src */ HGLYPH glyph, int srcx, int srcy,
1185                int fgcolour)                int bgcolour, int fgcolour)
1186  {  {
1187          Pixmap pixmap = (Pixmap) glyph;          SET_FOREGROUND(fgcolour);
1188            SET_BACKGROUND(bgcolour);
         xwin_set_function(ROP2_COPY);  
   
   
         XSetForeground(display, gc, Ctrans(fgcolour));  
         switch (mixmode)  
         {  
                 case MIX_TRANSPARENT:  
                         XSetStipple(display, gc, pixmap);  
                         XSetFillStyle(display, gc, FillStippled);  
                         XSetTSOrigin(display, gc, x, y);  
                         XFillRectangle(display, wnd, gc, x, y, cx, cy);  
                         XSetFillStyle(display, gc, FillSolid);  
                         break;  
   
                 case MIX_OPAQUE:  
                         XSetBackground(display, gc, Ctrans(bgcolour));  
 /*      XCopyPlane (display, pixmap, back_pixmap, back_gc, srcx, srcy, cx, cy, x, y, 1); */  
                         XSetStipple(display, gc, pixmap);  
                         XSetFillStyle(display, gc, FillOpaqueStippled);  
                         XSetTSOrigin(display, gc, x, y);  
                         XFillRectangle(display, wnd, gc, x, y, cx, cy);  
                         XSetFillStyle(display, gc, FillSolid);  
                         break;  
1189    
1190                  default:          XSetFillStyle(display, gc,
1191                          NOTIMP("mix %d\n", mixmode);                        (mixmode == MIX_TRANSPARENT) ? FillStippled : FillOpaqueStippled);
1192          }          XSetStipple(display, gc, (Pixmap) glyph);
1193            XSetTSOrigin(display, gc, x, y);
1194    
1195            FILL_RECTANGLE_FAST(x, y, cx, cy);
1196    
1197            XSetFillStyle(display, gc, FillSolid);
1198    }
1199    
1200    #define DO_GLYPH(ttext,idx) \
1201    {\
1202      glyph = cache_get_font (font, ttext[idx]);\
1203      if (!(flags & TEXT2_IMPLICIT_X))\
1204        {\
1205          xyoffset = ttext[++idx];\
1206          if ((xyoffset & 0x80))\
1207            {\
1208              if (flags & TEXT2_VERTICAL) \
1209                y += ttext[idx+1] | (ttext[idx+2] << 8);\
1210              else\
1211                x += ttext[idx+1] | (ttext[idx+2] << 8);\
1212              idx += 2;\
1213            }\
1214          else\
1215            {\
1216              if (flags & TEXT2_VERTICAL) \
1217                y += xyoffset;\
1218              else\
1219                x += xyoffset;\
1220            }\
1221        }\
1222      if (glyph != NULL)\
1223        {\
1224          ui_draw_glyph (mixmode, x + (short) glyph->offset,\
1225                         y + (short) glyph->baseline,\
1226                         glyph->width, glyph->height,\
1227                         glyph->pixmap, 0, 0, bgcolour, fgcolour);\
1228          if (flags & TEXT2_IMPLICIT_X)\
1229            x += glyph->width;\
1230        }\
1231  }  }
1232    
1233  void  void
1234  ui_draw_text(uint8 font, uint8 flags, int mixmode, int x, int y,  ui_draw_text(uint8 font, uint8 flags, int mixmode, int x, int y,
1235               int clipx, int clipy, int clipcx, int clipcy,               int clipx, int clipy, int clipcx, int clipcy,
1236               int boxx, int boxy, int boxcx, int boxcy,               int boxx, int boxy, int boxcx, int boxcy, int bgcolour,
1237               int bgcolour, int fgcolour, uint8 *text, uint8 length)               int fgcolour, uint8 * text, uint8 length)
1238  {  {
1239          FONTGLYPH *glyph;          FONTGLYPH *glyph;
1240          int i, xyoffset;          int i, j, xyoffset;
1241            DATABLOB *entry;
1242    
1243          xwin_set_function(ROP2_COPY);          SET_FOREGROUND(bgcolour);
         XSetForeground(display, gc, Ctrans(bgcolour));  
1244    
1245          if (boxcx > 1)          if (boxcx > 1)
1246                  XFillRectangle(display, wnd, gc, boxx, boxy, boxcx, boxcy);          {
1247                    FILL_RECTANGLE_FAST(boxx, boxy, boxcx, boxcy);
1248            }
1249          else if (mixmode == MIX_OPAQUE)          else if (mixmode == MIX_OPAQUE)
1250                  XFillRectangle(display, wnd, gc, clipx, clipy, clipcx, clipcy);          {
1251                    FILL_RECTANGLE_FAST(clipx, clipy, clipcx, clipcy);
1252            }
1253    
1254          /* Paint text, character by character */          /* Paint text, character by character */
1255          for (i = 0; i < length; i++)          for (i = 0; i < length;)
1256          {          {
1257                  glyph = cache_get_font(font, text[i]);                  switch (text[i])
   
                 if (!(flags & TEXT2_IMPLICIT_X))  
   
1258                  {                  {
1259                          xyoffset = text[++i];                          case 0xff:
1260                          if ((xyoffset & 0x80))                                  if (i + 2 < length)
1261                          {                                          cache_put_text(text[i + 1], text, text[i + 2]);
                                 if (flags & 0x04)       /* vertical text */  
                                         y += text[++i] | (text[++i] << 8);  
1262                                  else                                  else
1263                                          x += text[++i] | (text[++i] << 8);                                  {
1264                          }                                          error("this shouldn't be happening\n");
1265                          else                                          exit(1);
1266                          {                                  }
1267                                  if (flags & 0x04)       /* vertical text */                                  /* this will move pointer from start to first character after FF command */
1268                                          y += xyoffset;                                  length -= i + 3;
1269                                  else                                  text = &(text[i + 3]);
1270                                          x += xyoffset;                                  i = 0;
1271                          }                                  break;
1272    
1273                  }                          case 0xfe:
1274                  if (glyph != NULL)                                  entry = cache_get_text(text[i + 1]);
1275                  {                                  if (entry != NULL)
1276                          ui_draw_glyph(mixmode, x + (short) glyph->offset,                                  {
1277                                        y + (short) glyph->baseline,                                          if ((((uint8 *) (entry->data))[1] ==
1278                                        glyph->width, glyph->height,                                               0) && (!(flags & TEXT2_IMPLICIT_X)))
1279                                        glyph->pixmap, 0, 0,                                          {
1280                                        bgcolour, fgcolour);                                                  if (flags & TEXT2_VERTICAL)
1281                                                            y += text[i + 2];
1282                                                    else
1283                                                            x += text[i + 2];
1284                                            }
1285                                            for (j = 0; j < entry->size; j++)
1286                                                    DO_GLYPH(((uint8 *) (entry->data)), j);
1287                                    }
1288                                    if (i + 2 < length)
1289                                            i += 3;
1290                                    else
1291                                            i += 2;
1292                                    length -= i;
1293                                    /* this will move pointer from start to first character after FE command */
1294                                    text = &(text[i]);
1295                                    i = 0;
1296                                    break;
1297    
1298                          if (flags & TEXT2_IMPLICIT_X)                          default:
1299                                  x += glyph->width;                                  DO_GLYPH(text, i);
1300                                    i++;
1301                                    break;
1302                  }                  }
1303          }          }
1304            if (ownbackstore)
1305            {
1306                    if (boxcx > 1)
1307                            XCopyArea(display, backstore, wnd, gc, boxx,
1308                                      boxy, boxcx, boxcy, boxx, boxy);
1309                    else
1310                            XCopyArea(display, backstore, wnd, gc, clipx,
1311                                      clipy, clipcx, clipcy, clipx, clipy);
1312            }
1313  }  }
1314    
1315  void  void
# Line 887  ui_desktop_save(uint32 offset, int x, in Line 1318  ui_desktop_save(uint32 offset, int x, in
1318          Pixmap pix;          Pixmap pix;
1319          XImage *image;          XImage *image;
1320    
1321          pix = XCreatePixmap(display, wnd, cx, cy, depth);          if (ownbackstore)
1322          xwin_set_function(ROP2_COPY);          {
1323                    image = XGetImage(display, backstore, x, y, cx, cy, AllPlanes, ZPixmap);
1324          XCopyArea(display, wnd, pix, gc, x, y, cx, cy, 0, 0);          }
1325          image = XGetImage(display, pix, 0, 0, cx, cy, AllPlanes, ZPixmap);          else
1326            {
1327                    pix = XCreatePixmap(display, wnd, cx, cy, depth);
1328                    XCopyArea(display, wnd, pix, gc, x, y, cx, cy, 0, 0);
1329                    image = XGetImage(display, pix, 0, 0, cx, cy, AllPlanes, ZPixmap);
1330                    XFreePixmap(display, pix);
1331            }
1332    
1333          offset *= bpp/8;          offset *= bpp / 8;
1334          cache_put_desktop(offset, cx, cy, image->bytes_per_line,          cache_put_desktop(offset, cx, cy, image->bytes_per_line, bpp / 8, (uint8 *) image->data);
                           bpp/8, image->data);  
1335    
1336          XDestroyImage(image);          XDestroyImage(image);
         XFreePixmap(display, pix);  
1337  }  }
1338    
1339  void  void
# Line 907  ui_desktop_restore(uint32 offset, int x, Line 1342  ui_desktop_restore(uint32 offset, int x,
1342          XImage *image;          XImage *image;
1343          uint8 *data;          uint8 *data;
1344    
1345          offset *= bpp/8;          offset *= bpp / 8;
1346          data = cache_get_desktop(offset, cx, cy, bpp/8);          data = cache_get_desktop(offset, cx, cy, bpp / 8);
1347          if (data == NULL)          if (data == NULL)
1348                  return;                  return;
         image =  
                 XCreateImage(display, visual,  
                              depth, ZPixmap,  
                              0, data, cx, cy, BitmapPad(display),  
                              cx * bpp/8);  
         xwin_set_function(ROP2_COPY);  
         XPutImage(display, wnd, gc, image, 0, 0, x, y, cx, cy);  
         XFree(image);  
 }  
1349    
1350  /* unroll defines, used to make the loops a bit more readable... */          image = XCreateImage(display, visual, depth, ZPixmap, 0,
1351  #define unroll8Expr(uexp) uexp uexp uexp uexp uexp uexp uexp uexp                               (char *) data, cx, cy, BitmapPad(display), cx * bpp / 8);
 #define unroll8Lefts(uexp) case 7: uexp \  
         case 6: uexp \  
         case 5: uexp \  
         case 4: uexp \  
         case 3: uexp \  
         case 2: uexp \  
         case 1: uexp  
1352    
1353  static uint8 *          if (ownbackstore)
 translate(int width, int height, uint8 *data)  
 {  
         uint32 i;  
         uint32 size = width * height;  
         uint8 *d2 = xmalloc(size * bpp/8);  
         uint8 *d3 = d2;  
         uint32 pix;  
         i = (size & ~0x7);  
   
         /* XXX: where are the bits swapped??? */  
 #ifdef L_ENDIAN                 /* little-endian */  
         /* big-endian screen */  
         if (screen_msbfirst)  
1354          {          {
1355                  switch (bpp)                  XPutImage(display, backstore, gc, image, 0, 0, x, y, cx, cy);
1356                  {                  XCopyArea(display, backstore, wnd, gc, x, y, cx, cy, x, y);
                         case 32:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix >> 24;  
                                                     *d3++ = pix >> 16;  
                                                     *d3++ = pix >> 8;  
                                                     *d3++ = pix;) i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix >>  
                                                                      24;  
                                                                      *d3++ =  
                                                                      pix >>  
                                                                      16;  
                                                                      *d3++ =  
                                                                      pix >> 8;  
                                                                      *d3++ =  
                                                                      pix;)}  
                                 break;  
                         case 24:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix >> 16;  
                                                     *d3++ = pix >> 8;  
                                                     *d3++ = pix;) i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix >>  
                                                                      16;  
                                                                      *d3++ =  
                                                                      pix >> 8;  
                                                                      *d3++ =  
                                                                      pix;)}  
                                 break;  
                         case 16:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix >> 8;  
                                                     *d3++ = pix;) i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix >> 8;  
                                                                      *d3++ =  
                                                                      pix;)}  
                                 break;  
                         case 8:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix;  
                                                 )i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix;)}  
                                 break;  
                 }  
1357          }          }
1358          else          else
         {                       /* little-endian screen */  
                 switch (bpp)  
                 {  
                         case 32:  
                                 while (i)  
                                 {  
                                         unroll8Expr(*((uint32 *) d3) =  
                                                     colmap[*data++];  
                                                     d3 += sizeof(uint32);  
                                                 )i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(*  
                                                                      ((uint32  
                                                                        *) d3)  
 = colmap[*data++];  
 d3 += sizeof(uint32);  
                                         )}  
                                 break;  
                         case 24:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix;  
                                                     *d3++ = pix >> 8;  
                                                     *d3++ = pix >> 16;  
                                                 )i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix;  
                                                                      *d3++ =  
                                                                      pix >> 8;  
                                                                      *d3++ =  
                                                                      pix >>  
                                                                      16;)}  
                                 break;  
                         case 16:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix;  
                                                     *d3++ = pix >> 8;  
                                                 )i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix;  
                                                                      *d3++ =  
                                                                      pix >> 8;  
                                         )}  
                                 break;  
                         case 8:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix;  
                                                 )i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix;)}  
                 }  
         }  
   
 #else /* bigendian-compiled */  
         if (screen_msbfirst)  
1359          {          {
1360                  /* big-endian screen */                  XPutImage(display, wnd, gc, image, 0, 0, x, y, cx, cy);
                 switch (bpp)  
                 {  
                         case 32:  
                                 while (i)  
                                 {  
                                         unroll8Expr(*((uint32 *) d3) =  
                                                     colmap[*data++];  
                                                     d3 += sizeof(uint32);  
                                                 )i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(*  
                                                                      ((uint32  
                                                                        *) d3)  
 = colmap[*data++];  
 d3 += sizeof(uint32);  
                                         )}  
                                 break;  
                         case 24:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix;  
                                                     *d3++ = pix >> 8;  
                                                     *d3++ = pix >> 16;  
                                                 )i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix;  
                                                                      *d3++ =  
                                                                      pix >> 8;  
                                                                      *d3++ =  
                                                                      pix >>  
                                                                      16;)}  
                                 break;  
                         case 16:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix;  
                                                     *d3++ = pix >> 8;  
                                                 )i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix;  
                                                                      *d3++ =  
                                                                      pix >> 8;  
                                         )}  
                                 break;  
                         case 8:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix;  
                                                 )i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix;)}  
                 }  
1361          }          }
         else  
         {  
                 /* little-endian screen */  
                 switch (bpp)  
                 {  
                         case 32:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix;  
                                                     *d3++ = pix >> 8;  
                                                     *d3++ = pix >> 16;  
                                                     *d3++ = pix >> 24;  
                                                 )i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix;  
                                                                      *d3++ =  
                                                                      pix >> 8;  
                                                                      *d3++ =  
                                                                      pix >>  
                                                                      16;  
                                                                      *d3++ =  
                                                                      pix >>  
                                                                      24;)}  
                                 break;  
                         case 24:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix;  
                                                     *d3++ = pix >> 8;  
                                                     *d3++ = pix >> 16;  
                                                 )i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix;  
                                                                      *d3++ =  
                                                                      pix >> 8;  
                                                                      *d3++ =  
                                                                      pix >>  
                                                                      16;)}  
                                 break;  
                         case 16:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix;  
                                                     *d3++ = pix >> 8;  
                                                 )i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix;  
                                                                      *d3++ =  
                                                                      pix >> 8;  
                                         )}  
                                 break;  
                         case 8:  
                                 while (i)  
                                 {  
                                         unroll8Expr(pix = colmap[*data++];  
                                                     *d3++ = pix;  
                                                 )i -= 8;  
                                 }  
                                 i = (size & 0x7);  
                                 if (i != 0)  
                                         switch (i)  
                                         {  
                                                         unroll8Lefts(pix =  
                                                                      colmap  
                                                                      [*data++];  
                                                                      *d3++ =  
                                                                      pix;)}  
                 }  
         }  
 #endif  
1362    
1363          return d2;          XFree(image);
1364  }  }

Legend:
Removed from v.28  
changed lines
  Added in v.288

  ViewVC Help
Powered by ViewVC 1.1.26