842 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			842 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			C
		
	
	
	
| /**************************************************************
 | |
|  *
 | |
|  * Xquartz initialization code
 | |
|  *
 | |
|  * Copyright (c) 2007-2012 Apple Inc.
 | |
|  * Copyright (c) 2001-2004 Torrey T. Lyons. All Rights Reserved.
 | |
|  *
 | |
|  * Permission is hereby granted, free of charge, to any person obtaining a
 | |
|  * copy of this software and associated documentation files (the "Software"),
 | |
|  * to deal in the Software without restriction, including without limitation
 | |
|  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 | |
|  * and/or sell copies of the Software, and to permit persons to whom the
 | |
|  * Software is furnished to do so, subject to the following conditions:
 | |
|  *
 | |
|  * The above copyright notice and this permission notice shall be included in
 | |
|  * all copies or substantial portions of the Software.
 | |
|  *
 | |
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | |
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | |
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
 | |
|  * THE ABOVE LISTED COPYRIGHT HOLDER(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
 | |
|  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
 | |
|  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 | |
|  * DEALINGS IN THE SOFTWARE.
 | |
|  *
 | |
|  * Except as contained in this notice, the name(s) of the above copyright
 | |
|  * holders shall not be used in advertising or otherwise to promote the sale,
 | |
|  * use or other dealings in this Software without prior written authorization.
 | |
|  */
 | |
| 
 | |
| #include <dix-config.h>
 | |
| 
 | |
| #include <sys/stat.h>
 | |
| #include <X11/X.h>
 | |
| #include <X11/Xproto.h>
 | |
| 
 | |
| #include "os/ddx_priv.h"
 | |
| #include "os/osdep.h"
 | |
| 
 | |
| #include "os.h"
 | |
| #include "servermd.h"
 | |
| #include "inputstr.h"
 | |
| #include "scrnintstr.h"
 | |
| #include "mipointer.h"          // mi software cursor
 | |
| #include "micmap.h"             // mi colormap code
 | |
| #include "fb.h"                 // fb framebuffer code
 | |
| #include "globals.h"
 | |
| #include "dix.h"
 | |
| #include "xkbsrv.h"
 | |
| 
 | |
| #include <X11/extensions/XI.h>
 | |
| #include <X11/extensions/XIproto.h>
 | |
| #include "exevents.h"
 | |
| #include "extinit.h"
 | |
| #include "glx_extinit.h"
 | |
| #include "xserver-properties.h"
 | |
| 
 | |
| #include <sys/types.h>
 | |
| #include <sys/time.h>
 | |
| #include <sys/stat.h>
 | |
| #include <sys/syslimits.h>
 | |
| #include <stdio.h>
 | |
| #include <fcntl.h>
 | |
| #include <unistd.h>
 | |
| #include <stdarg.h>
 | |
| 
 | |
| #define HAS_UTSNAME 1
 | |
| #include <sys/utsname.h>
 | |
| 
 | |
| #define NO_CFPLUGIN
 | |
| #include <IOKit/hidsystem/IOHIDLib.h>
 | |
| 
 | |
| #include "input_priv.h"
 | |
| #include "screenint_priv.h"
 | |
| 
 | |
| #ifdef MITSHM
 | |
| #include "shmint.h"
 | |
| #endif
 | |
| 
 | |
| #include "darwin.h"
 | |
| #include "darwinEvents.h"
 | |
| #include "quartzKeyboard.h"
 | |
| #include "quartz.h"
 | |
| 
 | |
| #include "X11Application.h"
 | |
| 
 | |
| aslclient aslc;
 | |
| 
 | |
| void
 | |
| xq_asl_log(int level, const char *subsystem, const char *file,
 | |
|            const char *function, int line, const char *fmt,
 | |
|            ...)
 | |
| {
 | |
|     va_list args;
 | |
|     aslmsg msg = asl_new(ASL_TYPE_MSG);
 | |
| 
 | |
|     if (msg) {
 | |
|         char *_line;
 | |
| 
 | |
|         asl_set(msg, "File", file);
 | |
|         asl_set(msg, "Function", function);
 | |
|         asprintf(&_line, "%d", line);
 | |
|         if (_line) {
 | |
|             asl_set(msg, "Line", _line);
 | |
|             free(_line);
 | |
|         }
 | |
|         if (subsystem)
 | |
|             asl_set(msg, "Subsystem", subsystem);
 | |
|     }
 | |
| 
 | |
|     va_start(args, fmt);
 | |
|     asl_vlog(aslc, msg, level, fmt, args);
 | |
|     va_end(args);
 | |
| 
 | |
|     if (msg)
 | |
|         asl_free(msg);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * X server shared global variables
 | |
|  */
 | |
| int darwinScreensFound = 0;
 | |
| DevPrivateKeyRec darwinScreenKeyRec;
 | |
| io_connect_t darwinParamConnect = 0;
 | |
| int darwinEventReadFD = -1;
 | |
| int darwinEventWriteFD = -1;
 | |
| // int                     darwinMouseAccelChange = 1;
 | |
| int darwinFakeButtons = 0;
 | |
| 
 | |
| // location of X11's (0,0) point in global screen coordinates
 | |
| int darwinMainScreenX = 0;
 | |
| int darwinMainScreenY = 0;
 | |
| 
 | |
| // parameters read from the command line or user preferences
 | |
| int darwinDesiredDepth = -1;
 | |
| int darwinSyncKeymap = FALSE;
 | |
| 
 | |
| // modifier masks for faking mouse buttons - ANY of these bits trigger it  (not all)
 | |
| #ifdef NX_DEVICELCMDKEYMASK
 | |
| int darwinFakeMouse2Mask = NX_DEVICELALTKEYMASK | NX_DEVICERALTKEYMASK;
 | |
| int darwinFakeMouse3Mask = NX_DEVICELCMDKEYMASK | NX_DEVICERCMDKEYMASK;
 | |
| #else
 | |
| int darwinFakeMouse2Mask = NX_ALTERNATEMASK;
 | |
| int darwinFakeMouse3Mask = NX_COMMANDMASK;
 | |
| #endif
 | |
| 
 | |
| // Modifier mask for overriding event delivery to appkit (might be useful to set this to rcommand for input menu
 | |
| unsigned int darwinAppKitModMask = 0;            // Any of these bits
 | |
| 
 | |
| // Modifier mask for items in the Window menu (0 and -1 cause shortcuts to be disabled)
 | |
| unsigned int windowItemModMask = NX_COMMANDMASK;
 | |
| 
 | |
| // devices
 | |
| DeviceIntPtr darwinKeyboard = NULL;
 | |
| DeviceIntPtr darwinPointer = NULL;
 | |
| DeviceIntPtr darwinTabletStylus = NULL;
 | |
| DeviceIntPtr darwinTabletCursor = NULL;
 | |
| DeviceIntPtr darwinTabletEraser = NULL;
 | |
| 
 | |
| // Common pixmap formats
 | |
| static PixmapFormatRec formats[] = {
 | |
|     { 1,  1,  BITMAP_SCANLINE_PAD    },
 | |
|     { 4,  8,  BITMAP_SCANLINE_PAD    },
 | |
|     { 8,  8,  BITMAP_SCANLINE_PAD    },
 | |
|     { 15, 16, BITMAP_SCANLINE_PAD    },
 | |
|     { 16, 16, BITMAP_SCANLINE_PAD    },
 | |
|     { 24, 32, BITMAP_SCANLINE_PAD    },
 | |
|     { 32, 32, BITMAP_SCANLINE_PAD    }
 | |
| };
 | |
| 
 | |
| void
 | |
| DarwinPrintBanner(void)
 | |
| {
 | |
|     ErrorF("Xquartz starting:\n");
 | |
|     ErrorF("X.Org X Server %s\n", XSERVER_VERSION);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * DarwinScreenInit
 | |
|  *  This is a callback from dix during AddScreen() from InitOutput().
 | |
|  *  Initialize the screen and communicate information about it back to dix.
 | |
|  */
 | |
| static Bool
 | |
| DarwinScreenInit(ScreenPtr pScreen, int argc, char **argv)
 | |
| {
 | |
|     int dpi;
 | |
|     static int foundIndex = 0;
 | |
|     Bool ret;
 | |
|     DarwinFramebufferPtr dfb;
 | |
| 
 | |
|     if (!dixRegisterPrivateKey(&darwinScreenKeyRec, PRIVATE_SCREEN, 0))
 | |
|         return FALSE;
 | |
| 
 | |
|     // reset index of found screens for each server generation
 | |
|     if (pScreen->myNum == 0) {
 | |
|         foundIndex = 0;
 | |
| 
 | |
|         // reset the visual list
 | |
|         miClearVisualTypes();
 | |
|     }
 | |
| 
 | |
|     // allocate space for private per screen storage
 | |
|     dfb = malloc(sizeof(DarwinFramebufferRec));
 | |
| 
 | |
|     // SCREEN_PRIV(pScreen) = dfb;
 | |
|     dixSetPrivate(&pScreen->devPrivates, darwinScreenKey, dfb);
 | |
| 
 | |
|     // setup hardware/mode specific details
 | |
|     ret = QuartzAddScreen(foundIndex, pScreen);
 | |
|     foundIndex++;
 | |
|     if (!ret)
 | |
|         return FALSE;
 | |
| 
 | |
|     // setup a single visual appropriate for our pixel type
 | |
|     if (!miSetVisualTypesAndMasks(dfb->depth, dfb->visuals, dfb->bitsPerRGB,
 | |
|                                   dfb->preferredCVC, dfb->redMask,
 | |
|                                   dfb->greenMask, dfb->blueMask)) {
 | |
|         return FALSE;
 | |
|     }
 | |
| 
 | |
|     // TODO: Make PseudoColor visuals not suck in TrueColor mode
 | |
|     // if(dfb->depth > 8)
 | |
|     //    miSetVisualTypesAndMasks(8, PseudoColorMask, 8, PseudoColor, 0, 0, 0);
 | |
|     //
 | |
|     // TODO: Re-add support for 15bit
 | |
|     // if (dfb->depth > 15)
 | |
|     //    miSetVisualTypesAndMasks(15, TrueColorMask, 5, TrueColor,
 | |
|     //                             RM_ARGB(0, 5, 5, 5), GM_ARGB(0, 5, 5,
 | |
|     //                                                          5),
 | |
|     //                             BM_ARGB(0, 5, 5, 5));
 | |
|     if (dfb->depth > 24)
 | |
|         miSetVisualTypesAndMasks(24, TrueColorMask, 8, TrueColor,
 | |
|                                  RM_ARGB(0, 8, 8, 8), GM_ARGB(0, 8, 8,
 | |
|                                                               8),
 | |
|                                  BM_ARGB(0, 8, 8, 8));
 | |
| 
 | |
|     miSetPixmapDepths();
 | |
| 
 | |
|     // machine independent screen init
 | |
|     // setup _Screen structure in pScreen
 | |
|     if (monitorResolution)
 | |
|         dpi = monitorResolution;
 | |
|     else
 | |
|         dpi = 96;
 | |
| 
 | |
|     // initialize fb
 | |
|     if (!fbScreenInit(pScreen,
 | |
|                       dfb->framebuffer,                  // pointer to screen bitmap
 | |
|                       dfb->width, dfb->height,           // screen size in pixels
 | |
|                       dpi, dpi,                          // dots per inch
 | |
|                       dfb->pitch / (dfb->bitsPerPixel / 8), // pixel width of framebuffer
 | |
|                       dfb->bitsPerPixel)) {              // bits per pixel for screen
 | |
|         return FALSE;
 | |
|     }
 | |
| 
 | |
|     if (!fbPictureInit(pScreen, 0, 0)) {
 | |
|         return FALSE;
 | |
|     }
 | |
| 
 | |
| #ifdef MITSHM
 | |
|     ShmRegisterFbFuncs(pScreen);
 | |
| #endif
 | |
| 
 | |
|     // finish mode dependent screen setup including cursor support
 | |
|     if (!QuartzSetupScreen(pScreen->myNum, pScreen)) {
 | |
|         return FALSE;
 | |
|     }
 | |
| 
 | |
|     // create and install the default colormap and
 | |
|     // set pScreen->blackPixel / pScreen->white
 | |
|     if (!miCreateDefColormap(pScreen)) {
 | |
|         return FALSE;
 | |
|     }
 | |
| 
 | |
|     pScreen->x = dfb->x;
 | |
|     pScreen->y = dfb->y;
 | |
| 
 | |
|     /*    ErrorF("Screen %d added: %dx%d @ (%d,%d)\n",
 | |
|        index, dfb->width, dfb->height, dfb->x, dfb->y); */
 | |
| 
 | |
|     return TRUE;
 | |
| }
 | |
| 
 | |
| /*
 | |
|    =============================================================================
 | |
| 
 | |
|    mouse and keyboard callbacks
 | |
| 
 | |
|    =============================================================================
 | |
|  */
 | |
| 
 | |
| static void
 | |
| DarwinInputHandlerNotify(int fd __unused, int ready __unused, void *data __unused)
 | |
| {
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * DarwinMouseProc: Handle the initialization, etc. of a mouse
 | |
|  */
 | |
| static int
 | |
| DarwinMouseProc(DeviceIntPtr pPointer, int what)
 | |
| {
 | |
| #define NBUTTONS 3
 | |
| #define NAXES    6
 | |
|     // 3 buttons: left, middle, right
 | |
|     CARD8 map[NBUTTONS + 1] = { 0, 1, 2, 3};
 | |
|     Atom btn_labels[NBUTTONS] = { 0 };
 | |
|     Atom axes_labels[NAXES] = { 0 };
 | |
| 
 | |
|     switch (what) {
 | |
|     case DEVICE_INIT:
 | |
|         pPointer->public.on = FALSE;
 | |
| 
 | |
|         btn_labels[0] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_LEFT);
 | |
|         btn_labels[1] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_MIDDLE);
 | |
|         btn_labels[2] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_RIGHT);
 | |
| 
 | |
|         axes_labels[0] = XIGetKnownProperty(AXIS_LABEL_PROP_ABS_X);
 | |
|         axes_labels[1] = XIGetKnownProperty(AXIS_LABEL_PROP_ABS_Y);
 | |
|         axes_labels[2] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_X);
 | |
|         axes_labels[3] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_Y);
 | |
|         axes_labels[4] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_WHEEL);
 | |
|         axes_labels[5] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_HWHEEL);
 | |
| 
 | |
|         // Set button map.
 | |
|         InitPointerDeviceStruct((DevicePtr)pPointer, map, NBUTTONS,
 | |
|                                 btn_labels,
 | |
|                                 (PtrCtrlProcPtr)NoopDDA,
 | |
|                                 GetMotionHistorySize(), NAXES,
 | |
|                                 axes_labels);
 | |
|         InitValuatorAxisStruct(pPointer, 0, axes_labels[0],
 | |
|                                NO_AXIS_LIMITS, NO_AXIS_LIMITS,
 | |
|                                0, 0, 0, Absolute);
 | |
|         InitValuatorAxisStruct(pPointer, 1, axes_labels[1],
 | |
|                                NO_AXIS_LIMITS, NO_AXIS_LIMITS,
 | |
|                                0, 0, 0, Absolute);
 | |
|         InitValuatorAxisStruct(pPointer, 2, axes_labels[2],
 | |
|                                NO_AXIS_LIMITS, NO_AXIS_LIMITS,
 | |
|                                1, 0, 1, Relative);
 | |
|         InitValuatorAxisStruct(pPointer, 3, axes_labels[3],
 | |
|                                NO_AXIS_LIMITS, NO_AXIS_LIMITS,
 | |
|                                1, 0, 1, Relative);
 | |
|         InitValuatorAxisStruct(pPointer, 4, axes_labels[4],
 | |
|                                NO_AXIS_LIMITS, NO_AXIS_LIMITS,
 | |
|                                1, 0, 1, Relative);
 | |
|         InitValuatorAxisStruct(pPointer, 5, axes_labels[5],
 | |
|                                NO_AXIS_LIMITS, NO_AXIS_LIMITS,
 | |
|                                1, 0, 1, Relative);
 | |
| 
 | |
|         SetScrollValuator(pPointer, 4, SCROLL_TYPE_VERTICAL, -1.0, SCROLL_FLAG_PREFERRED);
 | |
|         SetScrollValuator(pPointer, 5, SCROLL_TYPE_HORIZONTAL, -1.0, SCROLL_FLAG_NONE);
 | |
|         break;
 | |
| 
 | |
|     case DEVICE_ON:
 | |
|         pPointer->public.on = TRUE;
 | |
|         SetNotifyFd(darwinEventReadFD, DarwinInputHandlerNotify, X_NOTIFY_READ, NULL);
 | |
|         return Success;
 | |
| 
 | |
|     case DEVICE_CLOSE:
 | |
|     case DEVICE_OFF:
 | |
|         pPointer->public.on = FALSE;
 | |
|         RemoveNotifyFd(darwinEventReadFD);
 | |
|         return Success;
 | |
|     }
 | |
| 
 | |
|     return Success;
 | |
| #undef NBUTTONS
 | |
| #undef NAXES
 | |
| }
 | |
| 
 | |
| static int
 | |
| DarwinTabletProc(DeviceIntPtr pPointer, int what)
 | |
| {
 | |
| #define NBUTTONS 3
 | |
| #define NAXES    5
 | |
|     CARD8 map[NBUTTONS + 1] = { 0, 1, 2, 3 };
 | |
|     Atom btn_labels[NBUTTONS] = { 0 };
 | |
|     Atom axes_labels[NAXES] = { 0 };
 | |
| 
 | |
|     switch (what) {
 | |
|     case DEVICE_INIT:
 | |
|         pPointer->public.on = FALSE;
 | |
| 
 | |
|         btn_labels[0] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_LEFT);
 | |
|         btn_labels[1] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_MIDDLE);
 | |
|         btn_labels[2] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_RIGHT);
 | |
| 
 | |
|         axes_labels[0] = XIGetKnownProperty(AXIS_LABEL_PROP_ABS_X);
 | |
|         axes_labels[1] = XIGetKnownProperty(AXIS_LABEL_PROP_ABS_Y);
 | |
|         axes_labels[2] = XIGetKnownProperty(AXIS_LABEL_PROP_ABS_PRESSURE);
 | |
|         axes_labels[3] = XIGetKnownProperty(AXIS_LABEL_PROP_ABS_TILT_X);
 | |
|         axes_labels[4] = XIGetKnownProperty(AXIS_LABEL_PROP_ABS_TILT_Y);
 | |
| 
 | |
|         // Set button map.
 | |
|         InitPointerDeviceStruct((DevicePtr)pPointer, map, NBUTTONS,
 | |
|                                 btn_labels,
 | |
|                                 (PtrCtrlProcPtr)NoopDDA,
 | |
|                                 GetMotionHistorySize(), NAXES,
 | |
|                                 axes_labels);
 | |
|         InitProximityClassDeviceStruct(pPointer);
 | |
| 
 | |
|         InitValuatorAxisStruct(pPointer, 0, axes_labels[0],
 | |
|                                0, XQUARTZ_VALUATOR_LIMIT,
 | |
|                                1, 0, 1, Absolute);
 | |
|         InitValuatorAxisStruct(pPointer, 1, axes_labels[1],
 | |
|                                0, XQUARTZ_VALUATOR_LIMIT,
 | |
|                                1, 0, 1, Absolute);
 | |
|         InitValuatorAxisStruct(pPointer, 2, axes_labels[2],
 | |
|                                0, XQUARTZ_VALUATOR_LIMIT,
 | |
|                                1, 0, 1, Absolute);
 | |
|         InitValuatorAxisStruct(pPointer, 3, axes_labels[3],
 | |
|                                -XQUARTZ_VALUATOR_LIMIT,
 | |
|                                XQUARTZ_VALUATOR_LIMIT,
 | |
|                                1, 0, 1, Absolute);
 | |
|         InitValuatorAxisStruct(pPointer, 4, axes_labels[4],
 | |
|                                -XQUARTZ_VALUATOR_LIMIT,
 | |
|                                XQUARTZ_VALUATOR_LIMIT,
 | |
|                                1, 0, 1, Absolute);
 | |
| 
 | |
|         //          pPointer->use = IsXExtensionDevice;
 | |
|         break;
 | |
| 
 | |
|     case DEVICE_ON:
 | |
|         pPointer->public.on = TRUE;
 | |
|         SetNotifyFd(darwinEventReadFD, DarwinInputHandlerNotify, X_NOTIFY_READ, NULL);
 | |
|         return Success;
 | |
| 
 | |
|     case DEVICE_CLOSE:
 | |
|     case DEVICE_OFF:
 | |
|         pPointer->public.on = FALSE;
 | |
|         RemoveNotifyFd(darwinEventReadFD);
 | |
|         return Success;
 | |
|     }
 | |
|     return Success;
 | |
| #undef NBUTTONS
 | |
| #undef NAXES
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * DarwinKeybdProc
 | |
|  *  Callback from X
 | |
|  */
 | |
| static int
 | |
| DarwinKeybdProc(DeviceIntPtr pDev, int onoff)
 | |
| {
 | |
|     switch (onoff) {
 | |
|     case DEVICE_INIT:
 | |
|         DarwinKeyboardInit(pDev);
 | |
|         break;
 | |
| 
 | |
|     case DEVICE_ON:
 | |
|         pDev->public.on = TRUE;
 | |
|         SetNotifyFd(darwinEventReadFD, DarwinInputHandlerNotify, X_NOTIFY_READ, NULL);
 | |
|         break;
 | |
| 
 | |
|     case DEVICE_OFF:
 | |
|         pDev->public.on = FALSE;
 | |
|         RemoveNotifyFd(darwinEventReadFD);
 | |
|         break;
 | |
| 
 | |
|     case DEVICE_CLOSE:
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| /*
 | |
|    ===========================================================================
 | |
| 
 | |
|    Utility routines
 | |
| 
 | |
|    ===========================================================================
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * DarwinParseModifierList
 | |
|  *  Parse a list of modifier names and return a corresponding modifier mask
 | |
|  */
 | |
| int
 | |
| DarwinParseModifierList(const char *constmodifiers, int separatelr)
 | |
| {
 | |
|     int result = 0;
 | |
| 
 | |
|     if (constmodifiers) {
 | |
|         char *modifiers = strdup(constmodifiers);
 | |
|         char *modifier;
 | |
|         int nxkey;
 | |
|         char *p = modifiers;
 | |
| 
 | |
|         while (p) {
 | |
|             modifier = strsep(&p, " ,+&|/"); // allow lots of separators
 | |
|             nxkey = DarwinModifierStringToNXMask(modifier, separatelr);
 | |
|             if (nxkey)
 | |
|                 result |= nxkey;
 | |
|             else
 | |
|                 ErrorF("fakebuttons: Unknown modifier \"%s\"\n", modifier);
 | |
|         }
 | |
|         free(modifiers);
 | |
|     }
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| /*
 | |
|    ===========================================================================
 | |
| 
 | |
|    Functions needed to link against device independent X
 | |
| 
 | |
|    ===========================================================================
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * InitInput
 | |
|  *  Register the keyboard and mouse devices
 | |
|  */
 | |
| void
 | |
| InitInput(int argc, char **argv)
 | |
| {
 | |
|     XkbRMLVOSet rmlvo = {
 | |
|         .rules   = "base", .model         = "empty", .layout = "empty",
 | |
|         .variant = NULL,   .options       = NULL
 | |
|     };
 | |
| 
 | |
|     /* We need to really have rules... or something... */
 | |
|     XkbSetRulesDflts(&rmlvo);
 | |
| 
 | |
|     assert(Success == AllocDevicePair(serverClient, "xquartz virtual",
 | |
|                                       &darwinPointer, &darwinKeyboard,
 | |
|                                       DarwinMouseProc, DarwinKeybdProc, FALSE));
 | |
| 
 | |
|     /* here's the snippet from the current gdk sources:
 | |
|        if (!strcmp (tmp_name, "pointer"))
 | |
|        gdkdev->info.source = GDK_SOURCE_MOUSE;
 | |
|        else if (!strcmp (tmp_name, "wacom") ||
 | |
|        !strcmp (tmp_name, "pen"))
 | |
|        gdkdev->info.source = GDK_SOURCE_PEN;
 | |
|        else if (!strcmp (tmp_name, "eraser"))
 | |
|        gdkdev->info.source = GDK_SOURCE_ERASER;
 | |
|        else if (!strcmp (tmp_name, "cursor"))
 | |
|        gdkdev->info.source = GDK_SOURCE_CURSOR;
 | |
|        else
 | |
|        gdkdev->info.source = GDK_SOURCE_PEN;
 | |
|      */
 | |
| 
 | |
|     darwinTabletStylus = AddInputDevice(serverClient, DarwinTabletProc, TRUE);
 | |
|     assert(darwinTabletStylus);
 | |
|     darwinTabletStylus->name = strdup("pen");
 | |
| 
 | |
|     darwinTabletCursor = AddInputDevice(serverClient, DarwinTabletProc, TRUE);
 | |
|     assert(darwinTabletCursor);
 | |
|     darwinTabletCursor->name = strdup("cursor");
 | |
| 
 | |
|     darwinTabletEraser = AddInputDevice(serverClient, DarwinTabletProc, TRUE);
 | |
|     assert(darwinTabletEraser);
 | |
|     darwinTabletEraser->name = strdup("eraser");
 | |
| 
 | |
|     DarwinEQInit();
 | |
| 
 | |
|     QuartzInitInput(argc, argv);
 | |
| }
 | |
| 
 | |
| void
 | |
| CloseInput(void)
 | |
| {
 | |
|     DarwinEQFini();
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * DarwinAdjustScreenOrigins
 | |
|  *  Shift all screens so the X11 (0, 0) coordinate is at the top
 | |
|  *  left of the global screen coordinates.
 | |
|  *
 | |
|  *  Screens can be arranged so the top left isn't on any screen, so
 | |
|  *  instead use the top left of the leftmost screen as (0,0). This
 | |
|  *  may mean some screen space is in -y, but it's better that (0,0)
 | |
|  *  be onscreen, or else default xterms disappear. It's better that
 | |
|  *  -y be used than -x, because when popup menus are forced
 | |
|  *  "onscreen" by dumb window managers like twm, they'll shift the
 | |
|  *  menus down instead of left, which still looks funny but is an
 | |
|  *  easier target to hit.
 | |
|  */
 | |
| void
 | |
| DarwinAdjustScreenOrigins(ScreenInfo *pScreenInfo)
 | |
| {
 | |
|     int i, left, top;
 | |
| 
 | |
|     left = pScreenInfo->screens[0]->x;
 | |
|     top = pScreenInfo->screens[0]->y;
 | |
| 
 | |
|     /* Find leftmost screen. If there's a tie, take the topmost of the two. */
 | |
|     for (i = 1; i < pScreenInfo->numScreens; i++) {
 | |
|         if (pScreenInfo->screens[i]->x < left ||
 | |
|             (pScreenInfo->screens[i]->x == left &&
 | |
|              pScreenInfo->screens[i]->y < top)) {
 | |
|             left = pScreenInfo->screens[i]->x;
 | |
|             top = pScreenInfo->screens[i]->y;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     darwinMainScreenX = left;
 | |
|     darwinMainScreenY = top;
 | |
| 
 | |
|     DEBUG_LOG("top = %d, left=%d\n", top, left);
 | |
| 
 | |
|     /* Shift all screens so that there is a screen whose top left
 | |
|      * is at X11 (0,0) and at global screen coordinate
 | |
|      * (darwinMainScreenX, darwinMainScreenY).
 | |
|      */
 | |
| 
 | |
|     if (darwinMainScreenX != 0 || darwinMainScreenY != 0) {
 | |
|         for (i = 0; i < pScreenInfo->numScreens; i++) {
 | |
|             pScreenInfo->screens[i]->x -= darwinMainScreenX;
 | |
|             pScreenInfo->screens[i]->y -= darwinMainScreenY;
 | |
|             DEBUG_LOG("Screen %d placed at X11 coordinate (%d,%d).\n",
 | |
|                       i, pScreenInfo->screens[i]->x,
 | |
|                       pScreenInfo->screens[i]->y);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* Update screenInfo.x/y */
 | |
|     update_desktop_dimensions();
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * InitOutput
 | |
|  *  Initialize screenInfo for all actually accessible framebuffers.
 | |
|  *
 | |
|  *  The display mode dependent code gets called three times. The mode
 | |
|  *  specific InitOutput routines are expected to discover the number
 | |
|  *  of potentially useful screens and cache routes to them internally.
 | |
|  *  Inside DarwinScreenInit are two other mode specific calls.
 | |
|  *  A mode specific AddScreen routine is called for each screen to
 | |
|  *  actually initialize the screen with the ScreenPtr structure.
 | |
|  *  After other screen setup has been done, a mode specific
 | |
|  *  SetupScreen function can be called to finalize screen setup.
 | |
|  */
 | |
| void
 | |
| InitOutput(ScreenInfo *pScreenInfo, int argc, char **argv)
 | |
| {
 | |
|     int i;
 | |
| 
 | |
|     pScreenInfo->imageByteOrder = IMAGE_BYTE_ORDER;
 | |
|     pScreenInfo->bitmapScanlineUnit = BITMAP_SCANLINE_UNIT;
 | |
|     pScreenInfo->bitmapScanlinePad = BITMAP_SCANLINE_PAD;
 | |
|     pScreenInfo->bitmapBitOrder = BITMAP_BIT_ORDER;
 | |
| 
 | |
|     // List how we want common pixmap formats to be padded
 | |
|     pScreenInfo->numPixmapFormats = ARRAY_SIZE(formats);
 | |
|     for (i = 0; i < ARRAY_SIZE(formats); i++)
 | |
|         pScreenInfo->formats[i] = formats[i];
 | |
| 
 | |
|     // Discover screens and do mode specific initialization
 | |
|     QuartzInitOutput(argc, argv);
 | |
| 
 | |
|     // Add screens
 | |
|     for (i = 0; i < darwinScreensFound; i++) {
 | |
|         AddScreen(DarwinScreenInit, argc, argv);
 | |
|     }
 | |
| 
 | |
|     xorgGlxCreateVendor();
 | |
| 
 | |
|     DarwinAdjustScreenOrigins(pScreenInfo);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * OsVendorFatalError
 | |
|  */
 | |
| void
 | |
| OsVendorFatalError(const char *f, va_list args)
 | |
| {
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * OsVendorInit
 | |
|  *  Initialization of Darwin OS support.
 | |
|  */
 | |
| void
 | |
| OsVendorInit(void)
 | |
| {
 | |
|     if (serverGeneration == 1) {
 | |
|         char *lf;
 | |
|         char *home = getenv("HOME");
 | |
|         assert(home);
 | |
|         assert(0 < asprintf(&lf, "%s/Library/Logs/X11", home));
 | |
| 
 | |
|         /* Ignore errors.  If EEXIST, we don't care.  If anything else,
 | |
|          * LogInit will handle it for us.
 | |
|          */
 | |
|         (void)mkdir(lf, S_IRWXU | S_IRWXG | S_IRWXO);
 | |
|         free(lf);
 | |
| 
 | |
|         assert(0 <
 | |
|                asprintf(&lf, "%s/Library/Logs/X11/%s.log", home,
 | |
|                         bundle_id_prefix));
 | |
|         LogInit(lf, ".old");
 | |
|         free(lf);
 | |
| 
 | |
|         DarwinPrintBanner();
 | |
|     }
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * ddxProcessArgument
 | |
|  *  Process device-dependent command line args. Returns 0 if argument is
 | |
|  *  not device dependent, otherwise Count of number of elements of argv
 | |
|  *  that are part of a device dependent commandline option.
 | |
|  */
 | |
| int
 | |
| ddxProcessArgument(int argc, char *argv[], int i)
 | |
| {
 | |
|     //    if ( !strcmp( argv[i], "-fullscreen" ) ) {
 | |
|     //        ErrorF( "Running full screen in parallel with Mac OS X Quartz window server.\n" );
 | |
|     //        return 1;
 | |
|     //    }
 | |
| 
 | |
|     //    if ( !strcmp( argv[i], "-rootless" ) ) {
 | |
|     //        ErrorF( "Running rootless inside Mac OS X window server.\n" );
 | |
|     //        return 1;
 | |
|     //    }
 | |
| 
 | |
|     // This command line arg is passed when launched from the Aqua GUI.
 | |
|     if (!strncmp(argv[i], "-psn_", 5)) {
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     if (!strcmp(argv[i], "-fakebuttons")) {
 | |
|         darwinFakeButtons = TRUE;
 | |
|         ErrorF("Faking a three button mouse\n");
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     if (!strcmp(argv[i], "-nofakebuttons")) {
 | |
|         darwinFakeButtons = FALSE;
 | |
|         ErrorF("Not faking a three button mouse\n");
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     if (!strcmp(argv[i], "-fakemouse2")) {
 | |
|         if (i == argc - 1) {
 | |
|             FatalError("-fakemouse2 must be followed by a modifier list\n");
 | |
|         }
 | |
|         if (!strcasecmp(argv[i + 1], "none") || !strcmp(argv[i + 1], ""))
 | |
|             darwinFakeMouse2Mask = 0;
 | |
|         else
 | |
|             darwinFakeMouse2Mask = DarwinParseModifierList(argv[i + 1], 1);
 | |
|         ErrorF("Modifier mask to fake mouse button 2 = 0x%x\n",
 | |
|                darwinFakeMouse2Mask);
 | |
|         return 2;
 | |
|     }
 | |
| 
 | |
|     if (!strcmp(argv[i], "-fakemouse3")) {
 | |
|         if (i == argc - 1) {
 | |
|             FatalError("-fakemouse3 must be followed by a modifier list\n");
 | |
|         }
 | |
|         if (!strcasecmp(argv[i + 1], "none") || !strcmp(argv[i + 1], ""))
 | |
|             darwinFakeMouse3Mask = 0;
 | |
|         else
 | |
|             darwinFakeMouse3Mask = DarwinParseModifierList(argv[i + 1], 1);
 | |
|         ErrorF("Modifier mask to fake mouse button 3 = 0x%x\n",
 | |
|                darwinFakeMouse3Mask);
 | |
|         return 2;
 | |
|     }
 | |
| 
 | |
|     if (!strcmp(argv[i], "+synckeymap")) {
 | |
|         darwinSyncKeymap = TRUE;
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     if (!strcmp(argv[i], "-synckeymap")) {
 | |
|         darwinSyncKeymap = FALSE;
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     if (!strcmp(argv[i], "-depth")) {
 | |
|         if (i == argc - 1) {
 | |
|             FatalError("-depth must be followed by a number\n");
 | |
|         }
 | |
|         darwinDesiredDepth = atoi(argv[i + 1]);
 | |
|         if (darwinDesiredDepth != -1 &&
 | |
|             darwinDesiredDepth != 8 &&
 | |
|             darwinDesiredDepth != 15 &&
 | |
|             darwinDesiredDepth != 24) {
 | |
|             FatalError("Unsupported pixel depth. Use 8, 15, or 24 bits\n");
 | |
|         }
 | |
| 
 | |
|         ErrorF("Attempting to use pixel depth of %i\n", darwinDesiredDepth);
 | |
|         return 2;
 | |
|     }
 | |
| 
 | |
|     if (!strcmp(argv[i], "-showconfig") || !strcmp(argv[i], "-version")) {
 | |
|         DarwinPrintBanner();
 | |
|         exit(0);
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * ddxUseMsg --
 | |
|  *  Print out correct use of device dependent commandline options.
 | |
|  *  Maybe the user now knows what really to do ...
 | |
|  */
 | |
| void
 | |
| ddxUseMsg(void)
 | |
| {
 | |
|     ErrorF("\n");
 | |
|     ErrorF("\n");
 | |
|     ErrorF("Device Dependent Usage:\n");
 | |
|     ErrorF("\n");
 | |
|     ErrorF("-depth <8,15,24> : use this bit depth.\n");
 | |
|     ErrorF(
 | |
|         "-fakebuttons : fake a three button mouse with Command and Option keys.\n");
 | |
|     ErrorF("-nofakebuttons : don't fake a three button mouse.\n");
 | |
|     ErrorF(
 | |
|         "-fakemouse2 <modifiers> : fake middle mouse button with modifier keys.\n");
 | |
|     ErrorF(
 | |
|         "-fakemouse3 <modifiers> : fake right mouse button with modifier keys.\n");
 | |
|     ErrorF(
 | |
|         "  ex: -fakemouse2 \"option,shift\" = option-shift-click is middle button.\n");
 | |
|     ErrorF("-version : show the server version.\n");
 | |
|     ErrorF("\n");
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * ddxGiveUp --
 | |
|  *      Device dependent cleanup. Called by dix before normal server death.
 | |
|  */
 | |
| void
 | |
| ddxGiveUp(enum ExitCode error)
 | |
| {
 | |
|     LogClose(error);
 | |
| }
 | |
| 
 | |
| #if INPUTTHREAD
 | |
| /** This function is called in Xserver/os/inputthread.c when starting
 | |
|     the input thread. */
 | |
| void
 | |
| ddxInputThreadInit(void)
 | |
| {
 | |
| }
 | |
| #endif
 |