Link Here
|
|
|
1 |
|
2 |
--- unix/xserver/hw/vnc/Input.cc 2011-11-08 21:44:10.000000000 +0900 |
3 |
+++ unix/xserver/hw/vnc/Input.cc 2013-05-27 18:54:41.184574000 +0900 |
4 |
@@ -82,10 +82,11 @@ |
5 |
/* Event queue is shared between all devices. */ |
6 |
#if XORG == 15 |
7 |
static xEvent *eventq = NULL; |
8 |
-#else |
9 |
+#elif XORG < 111 |
10 |
static EventList *eventq = NULL; |
11 |
#endif |
12 |
|
13 |
+#if XORG < 111 |
14 |
static void initEventq(void) |
15 |
{ |
16 |
/* eventq is never free()-ed because it exists during server life. */ |
17 |
@@ -100,7 +101,9 @@ |
18 |
#endif |
19 |
} |
20 |
} |
21 |
+#endif /* XORG < 111 */ |
22 |
|
23 |
+#if XORG < 111 |
24 |
static void enqueueEvents(DeviceIntPtr dev, int n) |
25 |
{ |
26 |
int i; |
27 |
@@ -122,6 +125,7 @@ |
28 |
); |
29 |
} |
30 |
} |
31 |
+#endif /* XORG < 111 */ |
32 |
|
33 |
InputDevice::InputDevice(rfb::VNCServerST *_server) |
34 |
: server(_server), oldButtonMask(0) |
35 |
@@ -141,12 +145,17 @@ |
36 |
keyboardProc, TRUE); |
37 |
RegisterKeyboardDevice(keyboardDev); |
38 |
#endif |
39 |
+#if XORG < 111 |
40 |
initEventq(); |
41 |
+#endif |
42 |
} |
43 |
|
44 |
void InputDevice::PointerButtonAction(int buttonMask) |
45 |
{ |
46 |
- int i, n; |
47 |
+ int i; |
48 |
+#if XORG < 111 |
49 |
+ int n; |
50 |
+#endif |
51 |
#if XORG >= 110 |
52 |
ValuatorMask mask; |
53 |
#endif |
54 |
@@ -160,13 +169,17 @@ |
55 |
#if XORG < 110 |
56 |
n = GetPointerEvents(eventq, pointerDev, action, i + 1, |
57 |
POINTER_RELATIVE, 0, 0, NULL); |
58 |
-#else |
59 |
+ enqueueEvents(pointerDev, n); |
60 |
+#elif XORG < 111 |
61 |
valuator_mask_set_range(&mask, 0, 0, NULL); |
62 |
n = GetPointerEvents(eventq, pointerDev, action, i + 1, |
63 |
POINTER_RELATIVE, &mask); |
64 |
-#endif |
65 |
enqueueEvents(pointerDev, n); |
66 |
- |
67 |
+#else |
68 |
+ valuator_mask_set_range(&mask, 0, 0, NULL); |
69 |
+ QueuePointerEvents(pointerDev, action, i + 1, |
70 |
+ POINTER_RELATIVE, &mask); |
71 |
+#endif |
72 |
} |
73 |
} |
74 |
|
75 |
@@ -175,7 +188,10 @@ |
76 |
|
77 |
void InputDevice::PointerMove(const rfb::Point &pos) |
78 |
{ |
79 |
- int n, valuators[2]; |
80 |
+ int valuators[2]; |
81 |
+#if XORG < 111 |
82 |
+ int n; |
83 |
+#endif |
84 |
#if XORG >= 110 |
85 |
ValuatorMask mask; |
86 |
#endif |
87 |
@@ -190,12 +206,16 @@ |
88 |
#if XORG < 110 |
89 |
n = GetPointerEvents(eventq, pointerDev, MotionNotify, 0, POINTER_ABSOLUTE, 0, |
90 |
2, valuators); |
91 |
-#else |
92 |
+ enqueueEvents(pointerDev, n); |
93 |
+#elif XORG < 111 |
94 |
valuator_mask_set_range(&mask, 0, 2, valuators); |
95 |
n = GetPointerEvents(eventq, pointerDev, MotionNotify, 0, POINTER_ABSOLUTE, |
96 |
&mask); |
97 |
-#endif |
98 |
enqueueEvents(pointerDev, n); |
99 |
+#else |
100 |
+ valuator_mask_set_range(&mask, 0, 2, valuators); |
101 |
+ QueuePointerEvents(pointerDev, MotionNotify, 0, POINTER_ABSOLUTE, &mask); |
102 |
+#endif |
103 |
|
104 |
cursorPos = pos; |
105 |
} |
106 |
@@ -298,14 +318,20 @@ |
107 |
static inline void pressKey(DeviceIntPtr dev, int kc, bool down, const char *msg) |
108 |
{ |
109 |
int action; |
110 |
+#if XORG < 111 |
111 |
unsigned int n; |
112 |
+#endif |
113 |
|
114 |
if (msg != NULL) |
115 |
vlog.debug("%s %d %s", msg, kc, down ? "down" : "up"); |
116 |
|
117 |
action = down ? KeyPress : KeyRelease; |
118 |
- n = GetKeyboardEvents(eventq, dev, action, kc); |
119 |
+#if XORG < 111 |
120 |
+ n = GetKeyboardEvents(eventq, dev, action, kc, NULL); |
121 |
enqueueEvents(dev, n); |
122 |
+#else |
123 |
+ QueueKeyboardEvents(dev, action, kc, NULL); |
124 |
+#endif |
125 |
} |
126 |
|
127 |
#define IS_PRESSED(keyc, keycode) \ |
128 |
@@ -340,8 +366,11 @@ |
129 |
int state, maxKeysPerMod, keycode; |
130 |
#if XORG >= 17 |
131 |
KeyCode *modmap = NULL; |
132 |
- |
133 |
+#if XORG >= 111 |
134 |
+ state = XkbStateFieldFromRec(&dev->master->key->xkbInfo->state); |
135 |
+#else /* XORG >= 111 */ |
136 |
state = XkbStateFieldFromRec(&dev->u.master->key->xkbInfo->state); |
137 |
+#endif /* XORG >= 111 */ |
138 |
#else |
139 |
KeyClassPtr keyc = dev->key; |
140 |
state = keyc->state; |
141 |
@@ -379,7 +408,11 @@ |
142 |
#if XORG >= 17 |
143 |
KeyCode *modmap = NULL; |
144 |
|
145 |
+#if XORG >= 111 |
146 |
+ keyc = dev->master->key; |
147 |
+#else /* XORG >= 111 */ |
148 |
keyc = dev->u.master->key; |
149 |
+#endif /* XORG >= 111 */ |
150 |
state = XkbStateFieldFromRec(&keyc->xkbInfo->state); |
151 |
#else |
152 |
keyc = dev->key; |
153 |
@@ -595,7 +628,11 @@ |
154 |
} |
155 |
|
156 |
#if XORG >= 17 |
157 |
+#if XORG >= 111 |
158 |
+ keyc = keyboardDev->master->key; |
159 |
+#else /* XORG >= 111 */ |
160 |
keyc = keyboardDev->u.master->key; |
161 |
+#endif /* XORG >= 111 */ |
162 |
|
163 |
keymap = XkbGetCoreMap(keyboardDev); |
164 |
if (!keymap) { |
165 |
@@ -752,7 +789,11 @@ |
166 |
XkbApplyMappingChange(keyboardDev, keymap, minKeyCode, |
167 |
maxKeyCode - minKeyCode + 1, |
168 |
NULL, serverClient); |
169 |
+#if XORG >= 111 |
170 |
+ XkbCopyDeviceKeymap(keyboardDev->master, keyboardDev); |
171 |
+#else |
172 |
XkbCopyDeviceKeymap(keyboardDev->u.master, keyboardDev); |
173 |
+#endif |
174 |
#endif /* XORG < 17 */ |
175 |
break; |
176 |
} |
177 |
|
178 |
--- unix/xserver/hw/vnc/Makefile.am 2011-10-31 17:14:40.000000000 +0900 |
179 |
+++ unix/xserver/hw/vnc/Makefile.am 2013-05-27 18:54:41.196574000 +0900 |
180 |
@@ -63,7 +63,7 @@ |
181 |
BUILT_SOURCES = $(nodist_Xvnc_SOURCES) |
182 |
|
183 |
fb.h: $(top_srcdir)/fb/fb.h |
184 |
- cat $(top_srcdir)/fb/fb.h | sed -e 's,and,c_and,' -e 's,xor,c_xor,' > $(srcdir)/fb.h |
185 |
+ cat $(top_srcdir)/fb/fb.h | sed -e 's,and,c_and,g' -e 's,xor,c_xor,g' > $(srcdir)/fb.h |
186 |
|
187 |
pixman.h: |
188 |
for i in ${XSERVERLIBS_CFLAGS}; do \ |
189 |
@@ -78,4 +78,4 @@ |
190 |
fi |
191 |
|
192 |
fbrop.h: $(top_srcdir)/fb/fbrop.h |
193 |
- cat $(top_srcdir)/fb/fbrop.h | sed -e 's,and,c_and,' -e 's,xor,c_xor,' > $(srcdir)/fbrop.h |
194 |
+ cat $(top_srcdir)/fb/fbrop.h | sed -e 's,and,c_and,g' -e 's,xor,c_xor,g' > $(srcdir)/fbrop.h |
195 |
|
196 |
--- unix/xserver/hw/vnc/XserverDesktop.cc 2012-01-24 00:54:11.000000000 +0900 |
197 |
+++ unix/xserver/hw/vnc/XserverDesktop.cc 2013-05-27 18:54:41.203576000 +0900 |
198 |
@@ -200,6 +200,8 @@ |
199 |
|
200 |
void XserverDesktop::setFramebuffer(int w, int h, void* fbptr, int stride) |
201 |
{ |
202 |
+ ScreenSet layout; |
203 |
+ |
204 |
width_ = w; |
205 |
height_ = h; |
206 |
|
207 |
@@ -217,9 +219,98 @@ |
208 |
data = (rdr::U8*)fbptr; |
209 |
stride_ = stride; |
210 |
|
211 |
- server->setPixelBuffer(this); |
212 |
+ layout = computeScreenLayout(); |
213 |
+ |
214 |
+ server->setPixelBuffer(this, layout); |
215 |
+} |
216 |
+ |
217 |
+void XserverDesktop::refreshScreenLayout() |
218 |
+{ |
219 |
+ server->setScreenLayout(computeScreenLayout()); |
220 |
+} |
221 |
+ |
222 |
+ScreenSet XserverDesktop::computeScreenLayout() |
223 |
+{ |
224 |
+ ScreenSet layout; |
225 |
+ |
226 |
+#ifdef RANDR |
227 |
+ rrScrPrivPtr rp = rrGetScrPriv(pScreen); |
228 |
+ OutputIdMap newIdMap; |
229 |
+ |
230 |
+ for (int i = 0;i < rp->numOutputs;i++) { |
231 |
+ RROutputPtr output; |
232 |
+ RRCrtcPtr crtc; |
233 |
+ |
234 |
+ output = rp->outputs[i]; |
235 |
+ crtc = output->crtc; |
236 |
+ |
237 |
+ /* Disabled? */ |
238 |
+ if ((crtc == NULL) || (crtc->mode == NULL)) |
239 |
+ continue; |
240 |
+ |
241 |
+ /* Known output? */ |
242 |
+ if (outputIdMap.count(output) == 1) |
243 |
+ newIdMap[output] = outputIdMap[output]; |
244 |
+ else { |
245 |
+ rdr::U32 id; |
246 |
+ OutputIdMap::const_iterator iter; |
247 |
+ |
248 |
+ while (true) { |
249 |
+ id = rand(); |
250 |
+ for (iter = outputIdMap.begin();iter != outputIdMap.end();++iter) { |
251 |
+ if (iter->second == id) |
252 |
+ break; |
253 |
+ } |
254 |
+ if (iter == outputIdMap.end()) |
255 |
+ break; |
256 |
+ } |
257 |
+ |
258 |
+ newIdMap[output] = id; |
259 |
+ } |
260 |
+ |
261 |
+ layout.add_screen(Screen(newIdMap[output], crtc->x, crtc->y, |
262 |
+ crtc->mode->mode.width, |
263 |
+ crtc->mode->mode.height, |
264 |
+ 0)); |
265 |
+ } |
266 |
+ |
267 |
+ /* Only keep the entries that are currently active */ |
268 |
+ outputIdMap = newIdMap; |
269 |
+#endif |
270 |
+ |
271 |
+ /* |
272 |
+ * Make sure we have something to display. Hopefully it's just temporary |
273 |
+ * that we have no active outputs... |
274 |
+ */ |
275 |
+ if (layout.num_screens() == 0) |
276 |
+ layout.add_screen(Screen(0, 0, 0, pScreen->width, pScreen->height, 0)); |
277 |
+ |
278 |
+ return layout; |
279 |
} |
280 |
|
281 |
+#ifdef RANDR |
282 |
+ |
283 |
+extern RRModePtr vncRandRModeGet(int width, int height); |
284 |
+ |
285 |
+RRModePtr XserverDesktop::findRandRMode(RROutputPtr output, int width, int height) |
286 |
+{ |
287 |
+ RRModePtr mode; |
288 |
+ |
289 |
+ for (int i = 0;i < output->numModes;i++) { |
290 |
+ if ((output->modes[i]->mode.width == width) && |
291 |
+ (output->modes[i]->mode.height == height)) |
292 |
+ return output->modes[i]; |
293 |
+ } |
294 |
+ |
295 |
+ mode = vncRandRModeGet(width, height); |
296 |
+ if (mode != NULL) |
297 |
+ return mode; |
298 |
+ |
299 |
+ return NULL; |
300 |
+} |
301 |
+ |
302 |
+#endif |
303 |
+ |
304 |
char* XserverDesktop::substitute(const char* varName) |
305 |
{ |
306 |
if (strcmp(varName, "$$") == 0) { |
307 |
@@ -727,100 +818,251 @@ |
308 |
vncClientCutText(str, len); |
309 |
} |
310 |
|
311 |
-#ifdef RANDR |
312 |
+extern RROutputPtr vncRandROutputCreate(ScreenPtr pScreen); |
313 |
+ |
314 |
unsigned int XserverDesktop::setScreenLayout(int fb_width, int fb_height, |
315 |
const rfb::ScreenSet& layout) |
316 |
{ |
317 |
- int i; |
318 |
- Bool ret; |
319 |
- RRScreenSizePtr pSize; |
320 |
- RROutputPtr output; |
321 |
- RRModePtr mode; |
322 |
- |
323 |
- // Make sure all RandR tables are properly populated |
324 |
-#if XORG == 15 |
325 |
- ret = RRGetInfo(pScreen); |
326 |
+#ifndef RANDR |
327 |
+ return rfb::resultProhibited; |
328 |
#else |
329 |
- ret = RRGetInfo(pScreen, FALSE); |
330 |
-#endif |
331 |
- if (!ret) |
332 |
- return resultNoResources; |
333 |
+ int availableOutputs; |
334 |
+ Bool ret; |
335 |
|
336 |
- // Register a new size, or get a reference to the existing one |
337 |
- pSize = RRRegisterSize(pScreen, fb_width, fb_height, |
338 |
- pScreen->mmWidth, pScreen->mmHeight); |
339 |
- if (!pSize) { |
340 |
- vlog.error("setScreenLayout: Could not get register new resolution"); |
341 |
- return resultNoResources; |
342 |
- } |
343 |
- ret = RRRegisterRate(pScreen, pSize, 60); |
344 |
- if (!ret) { |
345 |
- vlog.error("setScreenLayout: Could not register a rate for the resolution"); |
346 |
- return resultNoResources; |
347 |
- } |
348 |
- |
349 |
- // Then we have to call RRGetInfo again for it to copy the RandR |
350 |
- // 1.0 information to the 1.2 structures. |
351 |
-#if XORG == 15 |
352 |
- ret = RRGetInfo(pScreen); |
353 |
-#else |
354 |
- ret = RRGetInfo(pScreen, FALSE); |
355 |
-#endif |
356 |
- if (!ret) |
357 |
- return resultNoResources; |
358 |
+ rrScrPrivPtr rp = rrGetScrPriv(pScreen); |
359 |
|
360 |
- // Go via RandR to set the resolution in order for X11 notifications |
361 |
- // to be sent out properly. We currently only do RandR 1.0, but Xorg |
362 |
- // has dropped support for that API. So we have to emulate it via the |
363 |
- // same method ProcRRSetScreenConfig() uses. |
364 |
- // |
365 |
- // FIXME: This will cause setPixelBuffer() to be called, resulting in |
366 |
- // an unnecessary ExtendedDesktopSize to be sent. |
367 |
- |
368 |
- // We'll just reconfigure the first output |
369 |
- output = RRFirstOutput(pScreen); |
370 |
- if (!output) { |
371 |
- vlog.error("setScreenLayout: Could not get first output"); |
372 |
- return resultNoResources; |
373 |
- } |
374 |
- |
375 |
- // Find first mode with matching size |
376 |
- mode = NULL; |
377 |
- for (i = 0;i < output->numModes;i++) { |
378 |
- if ((output->modes[i]->mode.width == fb_width) && |
379 |
- (output->modes[i]->mode.height == fb_height)) { |
380 |
- mode = output->modes[i]; |
381 |
- break; |
382 |
+ /* |
383 |
+ * First check that we don't have any active clone modes. That's just |
384 |
+ * too messy to deal with. |
385 |
+ */ |
386 |
+ for (int i = 0;i < rp->numCrtcs;i++) { |
387 |
+ if (rp->crtcs[i]->numOutputs > 1) { |
388 |
+ vlog.error("Clone mode active. Refusing to touch screen layout."); |
389 |
+ return rfb::resultInvalid; |
390 |
+ } |
391 |
+ } |
392 |
+ |
393 |
+ /* |
394 |
+ * Next count how many useful outputs we have... |
395 |
+ * |
396 |
+ * This gets slightly complicated because we might need to hook a CRTC |
397 |
+ * up to the output, but also check that we don't try to use the same |
398 |
+ * CRTC for multiple outputs. |
399 |
+ */ |
400 |
+ std::set<RRCrtcPtr> usedCrtcs; |
401 |
+ availableOutputs = 0; |
402 |
+ for (int i = 0;i < rp->numOutputs;i++) { |
403 |
+ RROutputPtr output; |
404 |
+ |
405 |
+ output = rp->outputs[i]; |
406 |
+ |
407 |
+ if (output->crtc != NULL) |
408 |
+ availableOutputs++; |
409 |
+ else { |
410 |
+ for (int j = 0;j < output->numCrtcs;j++) { |
411 |
+ if (output->crtcs[j]->numOutputs != 0) |
412 |
+ continue; |
413 |
+ if (usedCrtcs.count(output->crtcs[j]) != 0) |
414 |
+ continue; |
415 |
+ |
416 |
+ availableOutputs++; |
417 |
+ usedCrtcs.insert(output->crtcs[j]); |
418 |
+ |
419 |
+ break; |
420 |
+ } |
421 |
} |
422 |
} |
423 |
- if (!mode) { |
424 |
- vlog.error("setScreenLayout: Could not find a matching mode"); |
425 |
- return resultNoResources; |
426 |
+ |
427 |
+ /* Try to create more outputs if needed... (only works on Xvnc) */ |
428 |
+ if (layout.num_screens() > availableOutputs) { |
429 |
+ for (int i = 0;i < (layout.num_screens() - availableOutputs);i++) { |
430 |
+ RROutputPtr output; |
431 |
+ output = vncRandROutputCreate(pScreen); |
432 |
+ if (output == NULL) { |
433 |
+ vlog.error("Unable to create more screens, as needed by the new client layout."); |
434 |
+ return rfb::resultInvalid; |
435 |
+ } |
436 |
+ } |
437 |
} |
438 |
|
439 |
- // Adjust screen size |
440 |
- ret = RRScreenSizeSet(pScreen, fb_width, fb_height, |
441 |
- pScreen->mmWidth, pScreen->mmHeight); |
442 |
- if (!ret) { |
443 |
- vlog.error("setScreenLayout: Could not adjust screen size"); |
444 |
- return resultNoResources; |
445 |
+ /* First we might need to resize the screen */ |
446 |
+ if ((fb_width != pScreen->width) || (fb_height != pScreen->height)) { |
447 |
+ /* Try to retain DPI when we resize */ |
448 |
+ ret = RRScreenSizeSet(pScreen, fb_width, fb_height, |
449 |
+ pScreen->mmWidth * fb_width / pScreen->width, |
450 |
+ pScreen->mmHeight * fb_height / pScreen->height); |
451 |
+ if (!ret) { |
452 |
+ vlog.error("Failed to resize screen to %dx%d", fb_width, fb_height); |
453 |
+ return rfb::resultInvalid; |
454 |
+ } |
455 |
+ } |
456 |
+ |
457 |
+ /* Next, reconfigure all known outputs, and turn off the other ones */ |
458 |
+ for (int i = 0;i < rp->numOutputs;i++) { |
459 |
+ RROutputPtr output; |
460 |
+ RRCrtcPtr crtc; |
461 |
+ RRModePtr mode; |
462 |
+ |
463 |
+ ScreenSet::const_iterator iter; |
464 |
+ |
465 |
+ output = rp->outputs[i]; |
466 |
+ crtc = output->crtc; |
467 |
+ |
468 |
+ /* Known? */ |
469 |
+ if (outputIdMap.count(output) == 0) |
470 |
+ continue; |
471 |
+ |
472 |
+ /* A known output should have a CRTC, but double check... */ |
473 |
+ if (crtc == NULL) { |
474 |
+ vlog.error("Existing output '%s' has unexpectedly been disabled", |
475 |
+ output->name); |
476 |
+ continue; |
477 |
+ } |
478 |
+ |
479 |
+ /* Find the corresponding screen... */ |
480 |
+ for (iter = layout.begin();iter != layout.end();++iter) { |
481 |
+ if (iter->id == outputIdMap[output]) |
482 |
+ break; |
483 |
+ } |
484 |
+ |
485 |
+ /* Missing? */ |
486 |
+ if (iter == layout.end()) { |
487 |
+ /* Disable and move on... */ |
488 |
+ ret = RRCrtcSet(crtc, NULL, crtc->x, crtc->y, crtc->rotation, |
489 |
+ crtc->numOutputs, crtc->outputs); |
490 |
+ if (!ret) { |
491 |
+ vlog.error("Failed to disable unused CRTC for output '%s'", |
492 |
+ output->name); |
493 |
+ return rfb::resultInvalid; |
494 |
+ } |
495 |
+ outputIdMap.erase(output); |
496 |
+ continue; |
497 |
+ } |
498 |
+ |
499 |
+ /* Need to switch mode? */ |
500 |
+ if ((crtc->mode->mode.width == iter->dimensions.width()) && |
501 |
+ (crtc->mode->mode.height == iter->dimensions.height())) |
502 |
+ mode = crtc->mode; |
503 |
+ else { |
504 |
+ mode = findRandRMode(output, iter->dimensions.width(), |
505 |
+ iter->dimensions.height()); |
506 |
+ if (mode == NULL) { |
507 |
+ vlog.error("Failed to find a suitable mode for %dx%d for output '%s'", |
508 |
+ iter->dimensions.width(), iter->dimensions.height(), |
509 |
+ output->name); |
510 |
+ return rfb::resultInvalid; |
511 |
+ } |
512 |
+ } |
513 |
+ |
514 |
+ /* Reconfigure new mode and position */ |
515 |
+ ret = RRCrtcSet(crtc, mode, iter->dimensions.tl.x, iter->dimensions.tl.y, |
516 |
+ crtc->rotation, crtc->numOutputs, crtc->outputs); |
517 |
+ if (!ret) { |
518 |
+ vlog.error("Failed to reconfigure output '%s' to %dx%d+%d+%d", |
519 |
+ output->name, |
520 |
+ iter->dimensions.width(), iter->dimensions.height(), |
521 |
+ iter->dimensions.tl.x, iter->dimensions.tl.y); |
522 |
+ return rfb::resultInvalid; |
523 |
+ } |
524 |
} |
525 |
|
526 |
- // And then the CRTC |
527 |
- ret = RRCrtcSet(output->crtc, mode, 0, 0, RR_Rotate_0, 1, &output); |
528 |
- if (!ret) { |
529 |
- vlog.error("setScreenLayout: Could not adjust CRTC"); |
530 |
- return resultNoResources; |
531 |
+ /* Finally, allocate new outputs for new screens */ |
532 |
+ ScreenSet::const_iterator iter; |
533 |
+ for (iter = layout.begin();iter != layout.end();++iter) { |
534 |
+ OutputIdMap::const_iterator oi; |
535 |
+ |
536 |
+ RROutputPtr output; |
537 |
+ RRCrtcPtr crtc; |
538 |
+ RRModePtr mode; |
539 |
+ |
540 |
+ int i; |
541 |
+ |
542 |
+ /* Does this screen have an output already? */ |
543 |
+ for (oi = outputIdMap.begin();oi != outputIdMap.end();++oi) { |
544 |
+ if (oi->second == iter->id) |
545 |
+ break; |
546 |
+ } |
547 |
+ |
548 |
+ if (oi != outputIdMap.end()) |
549 |
+ continue; |
550 |
+ |
551 |
+ /* Find an unused output */ |
552 |
+ for (i = 0;i < rp->numOutputs;i++) { |
553 |
+ output = rp->outputs[i]; |
554 |
+ crtc = output->crtc; |
555 |
+ |
556 |
+ /* In use? */ |
557 |
+ if (outputIdMap.count(output) == 1) |
558 |
+ continue; |
559 |
+ |
560 |
+ /* Need a CRTC? */ |
561 |
+ if (crtc == NULL) { |
562 |
+ for (int j = 0;j < output->numCrtcs;j++) { |
563 |
+ if (output->crtcs[j]->numOutputs != 0) |
564 |
+ continue; |
565 |
+ |
566 |
+ crtc = output->crtcs[j]; |
567 |
+ break; |
568 |
+ } |
569 |
+ |
570 |
+ /* Couldn't find one... */ |
571 |
+ if (crtc == NULL) |
572 |
+ continue; |
573 |
+ |
574 |
+ ret = RRCrtcSet(crtc, NULL, 0, 0, RR_Rotate_0, |
575 |
+ 1, &output); |
576 |
+ if (!ret) { |
577 |
+ vlog.error("Failed to associate a CRTC with output '%s'", |
578 |
+ output->name); |
579 |
+ return rfb::resultInvalid; |
580 |
+ } |
581 |
+ } |
582 |
+ |
583 |
+ break; |
584 |
+ } |
585 |
+ |
586 |
+ /* Shouldn't happen */ |
587 |
+ if (i == rp->numOutputs) |
588 |
+ return rfb::resultInvalid; |
589 |
+ |
590 |
+ mode = findRandRMode(output, iter->dimensions.width(), |
591 |
+ iter->dimensions.height()); |
592 |
+ if (mode == NULL) { |
593 |
+ vlog.error("Failed to find a suitable mode for %dx%d for output '%s'", |
594 |
+ iter->dimensions.width(), iter->dimensions.height(), |
595 |
+ output->name); |
596 |
+ return rfb::resultInvalid; |
597 |
+ } |
598 |
+ |
599 |
+ /* |
600 |
+ * Make sure we already have an entry for this, or |
601 |
+ * computeScreenLayout() will think it is a brand new output and |
602 |
+ * assign it a random id. |
603 |
+ */ |
604 |
+ outputIdMap[output] = iter->id; |
605 |
+ |
606 |
+ /* Reconfigure new mode and position */ |
607 |
+ ret = RRCrtcSet(crtc, mode, iter->dimensions.tl.x, iter->dimensions.tl.y, |
608 |
+ crtc->rotation, crtc->numOutputs, crtc->outputs); |
609 |
+ if (!ret) { |
610 |
+ vlog.error("Failed to reconfigure output '%s' to %dx%d+%d+%d", |
611 |
+ output->name, |
612 |
+ iter->dimensions.width(), iter->dimensions.height(), |
613 |
+ iter->dimensions.tl.x, iter->dimensions.tl.y); |
614 |
+ return rfb::resultInvalid; |
615 |
+ } |
616 |
} |
617 |
|
618 |
- // RandR 1.0 doesn't carry any screen layout information, so we need |
619 |
- // to update that manually. This results in another unnecessary |
620 |
- // ExtendedDesktopSize. |
621 |
- server->setScreenLayout(layout); |
622 |
+ /* |
623 |
+ * Update timestamp for when screen layout was last changed. |
624 |
+ * This is normally done in the X11 request handlers, which is |
625 |
+ * why we have to deal with it manually here. |
626 |
+ */ |
627 |
+ rp->lastSetTime = currentTime; |
628 |
|
629 |
- return resultSuccess; |
630 |
+ return rfb::resultSuccess; |
631 |
+#endif |
632 |
} |
633 |
-#endif // RANDR |
634 |
|
635 |
void XserverDesktop::grabRegion(const rfb::Region& region) |
636 |
{ |
637 |
|
638 |
--- unix/xserver/hw/vnc/XserverDesktop.h 2012-01-24 00:54:11.000000000 +0900 |
639 |
+++ unix/xserver/hw/vnc/XserverDesktop.h 2013-05-27 18:54:41.210573000 +0900 |
640 |
@@ -27,6 +27,8 @@ |
641 |
#include <dix-config.h> |
642 |
#endif |
643 |
|
644 |
+#include <map> |
645 |
+ |
646 |
#include <rfb/SDesktop.h> |
647 |
#include <rfb/HTTPServer.h> |
648 |
#include <rfb/PixelBuffer.h> |
649 |
@@ -39,6 +41,9 @@ |
650 |
#define class c_class |
651 |
#include <scrnintstr.h> |
652 |
#include <os.h> |
653 |
+#ifdef RANDR |
654 |
+#include <randrstr.h> |
655 |
+#endif |
656 |
#undef class |
657 |
} |
658 |
|
659 |
@@ -64,6 +69,7 @@ |
660 |
void blockUpdates(); |
661 |
void unblockUpdates(); |
662 |
void setFramebuffer(int w, int h, void* fbptr, int stride); |
663 |
+ void refreshScreenLayout(); |
664 |
void setColormap(ColormapPtr cmap); |
665 |
void setColourMapEntries(ColormapPtr pColormap, int ndef, xColorItem* pdef); |
666 |
void bell(); |
667 |
@@ -101,10 +107,8 @@ |
668 |
virtual void keyEvent(rdr::U32 key, bool down); |
669 |
virtual void clientCutText(const char* str, int len); |
670 |
virtual rfb::Point getFbSize() { return rfb::Point(width(), height()); } |
671 |
-#ifdef RANDR |
672 |
virtual unsigned int setScreenLayout(int fb_width, int fb_height, |
673 |
const rfb::ScreenSet& layout); |
674 |
-#endif |
675 |
|
676 |
// rfb::PixelBuffer callbacks |
677 |
virtual void grabRegion(const rfb::Region& r); |
678 |
@@ -123,6 +127,11 @@ |
679 |
|
680 |
private: |
681 |
void setColourMapEntries(int firstColour, int nColours); |
682 |
+ rfb::ScreenSet computeScreenLayout(); |
683 |
+#ifdef RANDR |
684 |
+ RRModePtr findRandRMode(RROutputPtr output, int width, int height); |
685 |
+#endif |
686 |
+ |
687 |
ScreenPtr pScreen; |
688 |
InputDevice *inputDevice; |
689 |
rfb::VNCServerST* server; |
690 |
@@ -139,5 +148,10 @@ |
691 |
void* queryConnectId; |
692 |
rfb::CharArray queryConnectAddress; |
693 |
rfb::CharArray queryConnectUsername; |
694 |
+ |
695 |
+#ifdef RANDR |
696 |
+ typedef std::map<RROutputPtr, rdr::U32> OutputIdMap; |
697 |
+ OutputIdMap outputIdMap; |
698 |
+#endif |
699 |
}; |
700 |
#endif |
701 |
|
702 |
--- unix/xserver/hw/vnc/vncExtInit.cc 2011-11-07 21:51:34.000000000 +0900 |
703 |
+++ unix/xserver/hw/vnc/vncExtInit.cc 2013-05-27 18:54:41.172575000 +0900 |
704 |
@@ -59,6 +59,7 @@ |
705 |
#include "XserverDesktop.h" |
706 |
#include "vncHooks.h" |
707 |
#include "vncExtInit.h" |
708 |
+#include "xorg-version.h" |
709 |
|
710 |
extern "C" { |
711 |
|
712 |
@@ -449,10 +450,16 @@ |
713 |
ev.window = cur->window; |
714 |
ev.time = GetTimeInMillis(); |
715 |
if (cur->client->swapped) { |
716 |
+#if XORG < 112 |
717 |
int n; |
718 |
swaps(&ev.sequenceNumber, n); |
719 |
swapl(&ev.window, n); |
720 |
swapl(&ev.time, n); |
721 |
+#else |
722 |
+ swaps(&ev.sequenceNumber); |
723 |
+ swapl(&ev.window); |
724 |
+ swapl(&ev.time); |
725 |
+#endif |
726 |
} |
727 |
WriteToClient(cur->client, sizeof(xVncExtClientCutTextNotifyEvent), |
728 |
(char *)&ev); |
729 |
@@ -495,9 +502,14 @@ |
730 |
ev.sequenceNumber = cur->client->sequence; |
731 |
ev.window = cur->window; |
732 |
if (cur->client->swapped) { |
733 |
+#if XORG < 112 |
734 |
int n; |
735 |
swaps(&ev.sequenceNumber, n); |
736 |
swapl(&ev.window, n); |
737 |
+#else |
738 |
+ swaps(&ev.sequenceNumber); |
739 |
+ swapl(&ev.window); |
740 |
+#endif |
741 |
} |
742 |
WriteToClient(cur->client, sizeof(xVncExtQueryConnectNotifyEvent), |
743 |
(char *)&ev); |
744 |
@@ -538,10 +550,16 @@ |
745 |
ev.window = cur->window; |
746 |
ev.selection = selection; |
747 |
if (cur->client->swapped) { |
748 |
+#if XORG < 112 |
749 |
int n; |
750 |
swaps(&ev.sequenceNumber, n); |
751 |
swapl(&ev.window, n); |
752 |
swapl(&ev.selection, n); |
753 |
+#else |
754 |
+ swaps(&ev.sequenceNumber); |
755 |
+ swapl(&ev.window); |
756 |
+ swapl(&ev.selection); |
757 |
+#endif |
758 |
} |
759 |
WriteToClient(cur->client, sizeof(xVncExtSelectionChangeNotifyEvent), |
760 |
(char *)&ev); |
761 |
@@ -562,7 +580,6 @@ |
762 |
param.buf[stuff->paramLen] = 0; |
763 |
|
764 |
xVncExtSetParamReply rep; |
765 |
- int n; |
766 |
rep.type = X_Reply; |
767 |
rep.length = 0; |
768 |
rep.success = 0; |
769 |
@@ -603,8 +620,14 @@ |
770 |
|
771 |
deny: |
772 |
if (client->swapped) { |
773 |
+#if XORG < 112 |
774 |
+ int n; |
775 |
swaps(&rep.sequenceNumber, n); |
776 |
swapl(&rep.length, n); |
777 |
+#else |
778 |
+ swaps(&rep.sequenceNumber); |
779 |
+ swapl(&rep.length); |
780 |
+#endif |
781 |
} |
782 |
WriteToClient(client, sizeof(xVncExtSetParamReply), (char *)&rep); |
783 |
return (client->noClientException); |
784 |
@@ -612,9 +635,13 @@ |
785 |
|
786 |
static int SProcVncExtSetParam(ClientPtr client) |
787 |
{ |
788 |
- register char n; |
789 |
REQUEST(xVncExtSetParamReq); |
790 |
+#if XORG < 112 |
791 |
+ register char n; |
792 |
swaps(&stuff->length, n); |
793 |
+#else |
794 |
+ swaps(&stuff->length); |
795 |
+#endif |
796 |
REQUEST_AT_LEAST_SIZE(xVncExtSetParamReq); |
797 |
return ProcVncExtSetParam(client); |
798 |
} |
799 |
@@ -628,7 +655,6 @@ |
800 |
param.buf[stuff->paramLen] = 0; |
801 |
|
802 |
xVncExtGetParamReply rep; |
803 |
- int n; |
804 |
rep.type = X_Reply; |
805 |
rep.sequenceNumber = client->sequence; |
806 |
rep.success = 0; |
807 |
@@ -646,9 +672,16 @@ |
808 |
rep.length = (len + 3) >> 2; |
809 |
rep.valueLen = len; |
810 |
if (client->swapped) { |
811 |
+#if XORG < 112 |
812 |
+ int n; |
813 |
swaps(&rep.sequenceNumber, n); |
814 |
swapl(&rep.length, n); |
815 |
swaps(&rep.valueLen, n); |
816 |
+#else |
817 |
+ swaps(&rep.sequenceNumber); |
818 |
+ swapl(&rep.length); |
819 |
+ swaps(&rep.valueLen); |
820 |
+#endif |
821 |
} |
822 |
WriteToClient(client, sizeof(xVncExtGetParamReply), (char *)&rep); |
823 |
if (value) |
824 |
@@ -659,9 +692,13 @@ |
825 |
|
826 |
static int SProcVncExtGetParam(ClientPtr client) |
827 |
{ |
828 |
- register char n; |
829 |
REQUEST(xVncExtGetParamReq); |
830 |
+#if XORG < 112 |
831 |
+ register char n; |
832 |
swaps(&stuff->length, n); |
833 |
+#else |
834 |
+ swaps(&stuff->length); |
835 |
+#endif |
836 |
REQUEST_AT_LEAST_SIZE(xVncExtGetParamReq); |
837 |
return ProcVncExtGetParam(client); |
838 |
} |
839 |
@@ -675,7 +712,6 @@ |
840 |
param.buf[stuff->paramLen] = 0; |
841 |
|
842 |
xVncExtGetParamDescReply rep; |
843 |
- int n; |
844 |
rep.type = X_Reply; |
845 |
rep.sequenceNumber = client->sequence; |
846 |
rep.success = 0; |
847 |
@@ -690,9 +726,16 @@ |
848 |
rep.length = (len + 3) >> 2; |
849 |
rep.descLen = len; |
850 |
if (client->swapped) { |
851 |
+#if XORG < 112 |
852 |
+ int n; |
853 |
swaps(&rep.sequenceNumber, n); |
854 |
swapl(&rep.length, n); |
855 |
swaps(&rep.descLen, n); |
856 |
+#else |
857 |
+ swaps(&rep.sequenceNumber); |
858 |
+ swapl(&rep.length); |
859 |
+ swaps(&rep.descLen); |
860 |
+#endif |
861 |
} |
862 |
WriteToClient(client, sizeof(xVncExtGetParamDescReply), (char *)&rep); |
863 |
if (desc) |
864 |
@@ -702,9 +745,13 @@ |
865 |
|
866 |
static int SProcVncExtGetParamDesc(ClientPtr client) |
867 |
{ |
868 |
- register char n; |
869 |
REQUEST(xVncExtGetParamDescReq); |
870 |
+#if XORG < 112 |
871 |
+ register char n; |
872 |
swaps(&stuff->length, n); |
873 |
+#else |
874 |
+ swaps(&stuff->length); |
875 |
+#endif |
876 |
REQUEST_AT_LEAST_SIZE(xVncExtGetParamDescReq); |
877 |
return ProcVncExtGetParamDesc(client); |
878 |
} |
879 |
@@ -715,7 +762,6 @@ |
880 |
REQUEST_SIZE_MATCH(xVncExtListParamsReq); |
881 |
|
882 |
xVncExtListParamsReply rep; |
883 |
- int n; |
884 |
rep.type = X_Reply; |
885 |
rep.sequenceNumber = client->sequence; |
886 |
|
887 |
@@ -731,9 +777,16 @@ |
888 |
rep.length = (len + 3) >> 2; |
889 |
rep.nParams = nParams; |
890 |
if (client->swapped) { |
891 |
+#if XORG < 112 |
892 |
+ int n; |
893 |
swaps(&rep.sequenceNumber, n); |
894 |
swapl(&rep.length, n); |
895 |
swaps(&rep.nParams, n); |
896 |
+#else |
897 |
+ swaps(&rep.sequenceNumber); |
898 |
+ swapl(&rep.length); |
899 |
+ swaps(&rep.nParams); |
900 |
+#endif |
901 |
} |
902 |
WriteToClient(client, sizeof(xVncExtListParamsReply), (char *)&rep); |
903 |
rdr::U8* data = new rdr::U8[len]; |
904 |
@@ -753,9 +806,13 @@ |
905 |
|
906 |
static int SProcVncExtListParams(ClientPtr client) |
907 |
{ |
908 |
- register char n; |
909 |
REQUEST(xVncExtListParamsReq); |
910 |
+#if XORG < 112 |
911 |
+ register char n; |
912 |
swaps(&stuff->length, n); |
913 |
+#else |
914 |
+ swaps(&stuff->length); |
915 |
+#endif |
916 |
REQUEST_SIZE_MATCH(xVncExtListParamsReq); |
917 |
return ProcVncExtListParams(client); |
918 |
} |
919 |
@@ -778,11 +835,19 @@ |
920 |
|
921 |
static int SProcVncExtSetServerCutText(ClientPtr client) |
922 |
{ |
923 |
- register char n; |
924 |
REQUEST(xVncExtSetServerCutTextReq); |
925 |
+#if XORG < 112 |
926 |
+ register char n; |
927 |
swaps(&stuff->length, n); |
928 |
+#else |
929 |
+ swaps(&stuff->length); |
930 |
+#endif |
931 |
REQUEST_AT_LEAST_SIZE(xVncExtSetServerCutTextReq); |
932 |
+#if XORG < 112 |
933 |
swapl(&stuff->textLen, n); |
934 |
+#else |
935 |
+ swapl(&stuff->textLen); |
936 |
+#endif |
937 |
return ProcVncExtSetServerCutText(client); |
938 |
} |
939 |
|
940 |
@@ -792,15 +857,21 @@ |
941 |
REQUEST_SIZE_MATCH(xVncExtGetClientCutTextReq); |
942 |
|
943 |
xVncExtGetClientCutTextReply rep; |
944 |
- int n; |
945 |
rep.type = X_Reply; |
946 |
rep.length = (clientCutTextLen + 3) >> 2; |
947 |
rep.sequenceNumber = client->sequence; |
948 |
rep.textLen = clientCutTextLen; |
949 |
if (client->swapped) { |
950 |
+#if XORG < 112 |
951 |
+ int n; |
952 |
swaps(&rep.sequenceNumber, n); |
953 |
swapl(&rep.length, n); |
954 |
swapl(&rep.textLen, n); |
955 |
+#else |
956 |
+ swaps(&rep.sequenceNumber); |
957 |
+ swapl(&rep.length); |
958 |
+ swapl(&rep.textLen); |
959 |
+#endif |
960 |
} |
961 |
WriteToClient(client, sizeof(xVncExtGetClientCutTextReply), (char *)&rep); |
962 |
if (clientCutText) |
963 |
@@ -810,9 +881,13 @@ |
964 |
|
965 |
static int SProcVncExtGetClientCutText(ClientPtr client) |
966 |
{ |
967 |
- register char n; |
968 |
REQUEST(xVncExtGetClientCutTextReq); |
969 |
+#if XORG < 112 |
970 |
+ register char n; |
971 |
swaps(&stuff->length, n); |
972 |
+#else |
973 |
+ swaps(&stuff->length); |
974 |
+#endif |
975 |
REQUEST_SIZE_MATCH(xVncExtGetClientCutTextReq); |
976 |
return ProcVncExtGetClientCutText(client); |
977 |
} |
978 |
@@ -842,12 +917,21 @@ |
979 |
|
980 |
static int SProcVncExtSelectInput(ClientPtr client) |
981 |
{ |
982 |
- register char n; |
983 |
REQUEST(xVncExtSelectInputReq); |
984 |
+#if XORG < 112 |
985 |
+ register char n; |
986 |
swaps(&stuff->length, n); |
987 |
+#else |
988 |
+ swaps(&stuff->length); |
989 |
+#endif |
990 |
REQUEST_SIZE_MATCH(xVncExtSelectInputReq); |
991 |
+#if XORG < 112 |
992 |
swapl(&stuff->window, n); |
993 |
swapl(&stuff->mask, n); |
994 |
+#else |
995 |
+ swapl(&stuff->window); |
996 |
+ swapl(&stuff->mask); |
997 |
+#endif |
998 |
return ProcVncExtSelectInput(client); |
999 |
} |
1000 |
|
1001 |
@@ -893,9 +977,14 @@ |
1002 |
rep.length = 0; |
1003 |
rep.sequenceNumber = client->sequence; |
1004 |
if (client->swapped) { |
1005 |
+#if XORG < 112 |
1006 |
int n; |
1007 |
swaps(&rep.sequenceNumber, n); |
1008 |
swapl(&rep.length, n); |
1009 |
+#else |
1010 |
+ swaps(&rep.sequenceNumber); |
1011 |
+ swapl(&rep.length); |
1012 |
+#endif |
1013 |
} |
1014 |
WriteToClient(client, sizeof(xVncExtConnectReply), (char *)&rep); |
1015 |
return (client->noClientException); |
1016 |
@@ -903,9 +992,13 @@ |
1017 |
|
1018 |
static int SProcVncExtConnect(ClientPtr client) |
1019 |
{ |
1020 |
- register char n; |
1021 |
REQUEST(xVncExtConnectReq); |
1022 |
+#if XORG < 112 |
1023 |
+ register char n; |
1024 |
swaps(&stuff->length, n); |
1025 |
+#else |
1026 |
+ swaps(&stuff->length); |
1027 |
+#endif |
1028 |
REQUEST_AT_LEAST_SIZE(xVncExtConnectReq); |
1029 |
return ProcVncExtConnect(client); |
1030 |
} |
1031 |
@@ -925,7 +1018,6 @@ |
1032 |
qcTimeout = 0; |
1033 |
|
1034 |
xVncExtGetQueryConnectReply rep; |
1035 |
- int n; |
1036 |
rep.type = X_Reply; |
1037 |
rep.sequenceNumber = client->sequence; |
1038 |
rep.timeout = qcTimeout; |
1039 |
@@ -934,11 +1026,20 @@ |
1040 |
rep.opaqueId = (CARD32)(long)queryConnectId; |
1041 |
rep.length = (rep.userLen + rep.addrLen + 3) >> 2; |
1042 |
if (client->swapped) { |
1043 |
+#if XORG < 112 |
1044 |
+ int n; |
1045 |
swaps(&rep.sequenceNumber, n); |
1046 |
swapl(&rep.userLen, n); |
1047 |
swapl(&rep.addrLen, n); |
1048 |
swapl(&rep.timeout, n); |
1049 |
swapl(&rep.opaqueId, n); |
1050 |
+#else |
1051 |
+ swaps(&rep.sequenceNumber); |
1052 |
+ swapl(&rep.userLen); |
1053 |
+ swapl(&rep.addrLen); |
1054 |
+ swapl(&rep.timeout); |
1055 |
+ swapl(&rep.opaqueId); |
1056 |
+#endif |
1057 |
} |
1058 |
WriteToClient(client, sizeof(xVncExtGetQueryConnectReply), (char *)&rep); |
1059 |
if (qcTimeout) |
1060 |
@@ -950,9 +1051,13 @@ |
1061 |
|
1062 |
static int SProcVncExtGetQueryConnect(ClientPtr client) |
1063 |
{ |
1064 |
- register char n; |
1065 |
REQUEST(xVncExtGetQueryConnectReq); |
1066 |
+#if XORG < 112 |
1067 |
+ register char n; |
1068 |
swaps(&stuff->length, n); |
1069 |
+#else |
1070 |
+ swaps(&stuff->length); |
1071 |
+#endif |
1072 |
REQUEST_SIZE_MATCH(xVncExtGetQueryConnectReq); |
1073 |
return ProcVncExtGetQueryConnect(client); |
1074 |
} |
1075 |
@@ -977,10 +1082,15 @@ |
1076 |
|
1077 |
static int SProcVncExtApproveConnect(ClientPtr client) |
1078 |
{ |
1079 |
- register char n; |
1080 |
REQUEST(xVncExtApproveConnectReq); |
1081 |
+#if XORG < 112 |
1082 |
+ register char n; |
1083 |
swaps(&stuff->length, n); |
1084 |
swapl(&stuff->opaqueId, n); |
1085 |
+#else |
1086 |
+ swaps(&stuff->length); |
1087 |
+ swapl(&stuff->opaqueId); |
1088 |
+#endif |
1089 |
REQUEST_SIZE_MATCH(xVncExtApproveConnectReq); |
1090 |
return ProcVncExtApproveConnect(client); |
1091 |
} |
1092 |
|
1093 |
--- unix/xserver/hw/vnc/vncHooks.cc 2012-01-24 00:54:11.000000000 +0900 |
1094 |
+++ unix/xserver/hw/vnc/vncHooks.cc 2013-05-27 18:55:18.837574000 +0900 |
1095 |
@@ -84,6 +84,8 @@ |
1096 |
#endif |
1097 |
#ifdef RANDR |
1098 |
RRSetConfigProcPtr RandRSetConfig; |
1099 |
+ RRScreenSetSizeProcPtr RandRScreenSetSize; |
1100 |
+ RRCrtcSetProcPtr RandRCrtcSet; |
1101 |
#endif |
1102 |
} vncHooksScreenRec, *vncHooksScreenPtr; |
1103 |
|
1104 |
@@ -143,6 +145,13 @@ |
1105 |
#ifdef RANDR |
1106 |
static Bool vncHooksRandRSetConfig(ScreenPtr pScreen, Rotation rotation, |
1107 |
int rate, RRScreenSizePtr pSize); |
1108 |
+static Bool vncHooksRandRScreenSetSize(ScreenPtr pScreen, |
1109 |
+ CARD16 width, CARD16 height, |
1110 |
+ CARD32 mmWidth, CARD32 mmHeight); |
1111 |
+static Bool vncHooksRandRCrtcSet(ScreenPtr pScreen, RRCrtcPtr crtc, |
1112 |
+ RRModePtr mode, int x, int y, |
1113 |
+ Rotation rotation, int numOutputs, |
1114 |
+ RROutputPtr *outputs); |
1115 |
#endif |
1116 |
|
1117 |
// GC "funcs" |
1118 |
@@ -283,6 +292,8 @@ |
1119 |
rp = rrGetScrPriv(pScreen); |
1120 |
if (rp) { |
1121 |
vncHooksScreen->RandRSetConfig = rp->rrSetConfig; |
1122 |
+ vncHooksScreen->RandRScreenSetSize = rp->rrScreenSetSize; |
1123 |
+ vncHooksScreen->RandRCrtcSet = rp->rrCrtcSet; |
1124 |
} |
1125 |
#endif |
1126 |
|
1127 |
@@ -304,7 +315,13 @@ |
1128 |
#endif |
1129 |
#ifdef RANDR |
1130 |
if (rp) { |
1131 |
- rp->rrSetConfig = vncHooksRandRSetConfig; |
1132 |
+ /* Some RandR callbacks are optional */ |
1133 |
+ if (rp->rrSetConfig) |
1134 |
+ rp->rrSetConfig = vncHooksRandRSetConfig; |
1135 |
+ if (rp->rrScreenSetSize) |
1136 |
+ rp->rrScreenSetSize = vncHooksRandRScreenSetSize; |
1137 |
+ if (rp->rrCrtcSet) |
1138 |
+ rp->rrCrtcSet = vncHooksRandRCrtcSet; |
1139 |
} |
1140 |
#endif |
1141 |
|
1142 |
@@ -361,6 +378,8 @@ |
1143 |
rp = rrGetScrPriv(pScreen); |
1144 |
if (rp) { |
1145 |
rp->rrSetConfig = vncHooksScreen->RandRSetConfig; |
1146 |
+ rp->rrScreenSetSize = vncHooksScreen->RandRScreenSetSize; |
1147 |
+ rp->rrCrtcSet = vncHooksScreen->RandRCrtcSet; |
1148 |
} |
1149 |
#endif |
1150 |
|
1151 |
@@ -596,42 +615,106 @@ |
1152 |
|
1153 |
#ifdef RANDR |
1154 |
|
1155 |
+static void vncPreScreenResize(ScreenPtr pScreen) |
1156 |
+{ |
1157 |
+ vncHooksScreenPtr vncHooksScreen = vncHooksScreenPrivate(pScreen); |
1158 |
+ |
1159 |
+ // We need to prevent the RFB core from accessing the framebuffer |
1160 |
+ // for a while as there might be updates thrown our way inside |
1161 |
+ // the routines that change the screen (i.e. before we have a |
1162 |
+ // pointer to the new framebuffer). |
1163 |
+ vncHooksScreen->desktop->blockUpdates(); |
1164 |
+} |
1165 |
+ |
1166 |
+static void vncPostScreenResize(ScreenPtr pScreen, Bool success) |
1167 |
+{ |
1168 |
+ vncHooksScreenPtr vncHooksScreen = vncHooksScreenPrivate(pScreen); |
1169 |
+ |
1170 |
+ RegionRec reg; |
1171 |
+ BoxRec box; |
1172 |
+ |
1173 |
+ if (success) { |
1174 |
+ // Let the RFB core know of the new dimensions and framebuffer |
1175 |
+ vncHooksScreen->desktop->setFramebuffer(pScreen->width, pScreen->height, |
1176 |
+ vncFbptr[pScreen->myNum], |
1177 |
+ vncFbstride[pScreen->myNum]); |
1178 |
+ } |
1179 |
+ |
1180 |
+ vncHooksScreen->desktop->unblockUpdates(); |
1181 |
+ |
1182 |
+ if (success) { |
1183 |
+ // Mark entire screen as changed |
1184 |
+ box.x1 = 0; |
1185 |
+ box.y1 = 0; |
1186 |
+ box.x2 = pScreen->width; |
1187 |
+ box.y2 = pScreen->height; |
1188 |
+ REGION_INIT(pScreen, ®, &box, 1); |
1189 |
+ |
1190 |
+ vncHooksScreen->desktop->add_changed(®); |
1191 |
+ } |
1192 |
+} |
1193 |
+ |
1194 |
static Bool vncHooksRandRSetConfig(ScreenPtr pScreen, Rotation rotation, |
1195 |
int rate, RRScreenSizePtr pSize) |
1196 |
{ |
1197 |
vncHooksScreenPtr vncHooksScreen = vncHooksScreenPrivate(pScreen); |
1198 |
rrScrPrivPtr rp = rrGetScrPriv(pScreen); |
1199 |
Bool ret; |
1200 |
- RegionRec reg; |
1201 |
- BoxRec box; |
1202 |
|
1203 |
- // We need to prevent the RFB core from accessing the framebuffer |
1204 |
- // for a while as there might be updates thrown our way inside |
1205 |
- // rrSetConfig (i.e. before we have a pointer to the new framebuffer). |
1206 |
- vncHooksScreen->desktop->blockUpdates(); |
1207 |
+ vncPreScreenResize(pScreen); |
1208 |
|
1209 |
rp->rrSetConfig = vncHooksScreen->RandRSetConfig; |
1210 |
ret = (*rp->rrSetConfig)(pScreen, rotation, rate, pSize); |
1211 |
rp->rrSetConfig = vncHooksRandRSetConfig; |
1212 |
|
1213 |
+ vncPostScreenResize(pScreen, ret); |
1214 |
+ |
1215 |
if (!ret) |
1216 |
return FALSE; |
1217 |
|
1218 |
- // Let the RFB core know of the new dimensions and framebuffer |
1219 |
- vncHooksScreen->desktop->setFramebuffer(pScreen->width, pScreen->height, |
1220 |
- vncFbptr[pScreen->myNum], |
1221 |
- vncFbstride[pScreen->myNum]); |
1222 |
+ return TRUE; |
1223 |
+} |
1224 |
|
1225 |
- vncHooksScreen->desktop->unblockUpdates(); |
1226 |
+static Bool vncHooksRandRScreenSetSize(ScreenPtr pScreen, |
1227 |
+ CARD16 width, CARD16 height, |
1228 |
+ CARD32 mmWidth, CARD32 mmHeight) |
1229 |
+{ |
1230 |
+ vncHooksScreenPtr vncHooksScreen = vncHooksScreenPrivate(pScreen); |
1231 |
+ rrScrPrivPtr rp = rrGetScrPriv(pScreen); |
1232 |
+ Bool ret; |
1233 |
+ |
1234 |
+ vncPreScreenResize(pScreen); |
1235 |
+ |
1236 |
+ rp->rrScreenSetSize = vncHooksScreen->RandRScreenSetSize; |
1237 |
+ ret = (*rp->rrScreenSetSize)(pScreen, width, height, mmWidth, mmHeight); |
1238 |
+ rp->rrScreenSetSize = vncHooksRandRScreenSetSize; |
1239 |
+ |
1240 |
+ vncPostScreenResize(pScreen, ret); |
1241 |
+ |
1242 |
+ if (!ret) |
1243 |
+ return FALSE; |
1244 |
+ |
1245 |
+ return TRUE; |
1246 |
+} |
1247 |
+ |
1248 |
+static Bool vncHooksRandRCrtcSet(ScreenPtr pScreen, RRCrtcPtr crtc, |
1249 |
+ RRModePtr mode, int x, int y, |
1250 |
+ Rotation rotation, int num_outputs, |
1251 |
+ RROutputPtr *outputs) |
1252 |
+{ |
1253 |
+ vncHooksScreenPtr vncHooksScreen = vncHooksScreenPrivate(pScreen); |
1254 |
+ rrScrPrivPtr rp = rrGetScrPriv(pScreen); |
1255 |
+ Bool ret; |
1256 |
|
1257 |
- // Mark entire screen as changed |
1258 |
- box.x1 = 0; |
1259 |
- box.y1 = 0; |
1260 |
- box.x2 = pScreen->width; |
1261 |
- box.y2 = pScreen->height; |
1262 |
- REGION_INIT(pScreen, ®, &box, 1); |
1263 |
+ rp->rrCrtcSet = vncHooksScreen->RandRCrtcSet; |
1264 |
+ ret = (*rp->rrCrtcSet)(pScreen, crtc, mode, x, y, rotation, |
1265 |
+ num_outputs, outputs); |
1266 |
+ rp->rrCrtcSet = vncHooksRandRCrtcSet; |
1267 |
|
1268 |
- vncHooksScreen->desktop->add_changed(®); |
1269 |
+ if (!ret) |
1270 |
+ return FALSE; |
1271 |
+ |
1272 |
+ vncHooksScreen->desktop->refreshScreenLayout(); |
1273 |
|
1274 |
return TRUE; |
1275 |
} |
1276 |
@@ -667,7 +750,7 @@ |
1277 |
}; |
1278 |
|
1279 |
|
1280 |
-// ValidateGC - wrap the "ops" if a viewable window |
1281 |
+// ValidateGC - wrap the "ops" if a viewable window OR the screen pixmap |
1282 |
|
1283 |
static void vncHooksValidateGC(GCPtr pGC, unsigned long changes, |
1284 |
DrawablePtr pDrawable) |
1285 |
@@ -679,7 +762,9 @@ |
1286 |
(*pGC->funcs->ValidateGC) (pGC, changes, pDrawable); |
1287 |
|
1288 |
u.vncHooksGC->wrappedOps = 0; |
1289 |
- if (pDrawable->type == DRAWABLE_WINDOW && ((WindowPtr) pDrawable)->viewable) { |
1290 |
+ if ((pDrawable->type == DRAWABLE_WINDOW && |
1291 |
+ ((WindowPtr) pDrawable)->viewable) || |
1292 |
+ (pDrawable == &pGC->pScreen->GetScreenPixmap(pGC->pScreen)->drawable)) { |
1293 |
u.vncHooksGC->wrappedOps = pGC->ops; |
1294 |
DBGPRINT((stderr,"vncHooksValidateGC: wrapped GC ops\n")); |
1295 |
} |
1296 |
@@ -750,7 +835,7 @@ |
1297 |
DBGPRINT((stderr,"vncHooks" #name " called\n")); |
1298 |
|
1299 |
|
1300 |
-// FillSpans - changed region is the whole of borderClip. This is pessimistic, |
1301 |
+// FillSpans - assume the entire clip region is damaged. This is pessimistic, |
1302 |
// but I believe this function is rarely used so it doesn't matter. |
1303 |
|
1304 |
static void vncHooksFillSpans(DrawablePtr pDrawable, GCPtr pGC, int nInit, |
1305 |
@@ -759,14 +844,18 @@ |
1306 |
{ |
1307 |
GC_OP_UNWRAPPER(pDrawable, pGC, FillSpans); |
1308 |
|
1309 |
- RegionHelper changed(pScreen, &((WindowPtr)pDrawable)->borderClip); |
1310 |
+ RegionHelper changed(pScreen, pGC->pCompositeClip); |
1311 |
+ |
1312 |
+ if (pDrawable->type == DRAWABLE_WINDOW) |
1313 |
+ REGION_INTERSECT(pScreen, changed.reg, changed.reg, |
1314 |
+ &((WindowPtr)pDrawable)->borderClip); |
1315 |
|
1316 |
(*pGC->ops->FillSpans) (pDrawable, pGC, nInit, pptInit, pwidthInit, fSorted); |
1317 |
|
1318 |
vncHooksScreen->desktop->add_changed(changed.reg); |
1319 |
} |
1320 |
|
1321 |
-// SetSpans - changed region is the whole of borderClip. This is pessimistic, |
1322 |
+// SetSpans - assume the entire clip region is damaged. This is pessimistic, |
1323 |
// but I believe this function is rarely used so it doesn't matter. |
1324 |
|
1325 |
static void vncHooksSetSpans(DrawablePtr pDrawable, GCPtr pGC, char *psrc, |
1326 |
@@ -775,7 +864,11 @@ |
1327 |
{ |
1328 |
GC_OP_UNWRAPPER(pDrawable, pGC, SetSpans); |
1329 |
|
1330 |
- RegionHelper changed(pScreen, &((WindowPtr)pDrawable)->borderClip); |
1331 |
+ RegionHelper changed(pScreen, pGC->pCompositeClip); |
1332 |
+ |
1333 |
+ if (pDrawable->type == DRAWABLE_WINDOW) |
1334 |
+ REGION_INTERSECT(pScreen, changed.reg, changed.reg, |
1335 |
+ &((WindowPtr)pDrawable)->borderClip); |
1336 |
|
1337 |
(*pGC->ops->SetSpans) (pDrawable, pGC, psrc, ppt, pwidth, nspans, fSorted); |
1338 |
|
1339 |
@@ -827,16 +920,23 @@ |
1340 |
|
1341 |
RegionHelper src(pScreen); |
1342 |
|
1343 |
- if ((pSrc->type == DRAWABLE_WINDOW) && (pSrc->pScreen == pScreen)) { |
1344 |
+ // The source of the data has to be something that's on screen. |
1345 |
+ // This means either a window, or the screen pixmap. |
1346 |
+ if ((pSrc->pScreen == pScreen) && |
1347 |
+ ((pSrc->type == DRAWABLE_WINDOW) || |
1348 |
+ (pSrc == &pScreen->GetScreenPixmap(pScreen)->drawable))) { |
1349 |
box.x1 = srcx + pSrc->x; |
1350 |
box.y1 = srcy + pSrc->y; |
1351 |
box.x2 = box.x1 + w; |
1352 |
box.y2 = box.y1 + h; |
1353 |
|
1354 |
src.init(&box, 0); |
1355 |
- if (REGION_NOTEMPTY(pScreen, &((WindowPtr)pSrc)->clipList)) { |
1356 |
- REGION_INTERSECT(pScreen, src.reg, src.reg, &((WindowPtr)pSrc)->clipList); |
1357 |
+ |
1358 |
+ if ((pSrc->type == DRAWABLE_WINDOW) && |
1359 |
+ REGION_NOTEMPTY(pScreen, &((WindowPtr)pSrc)->clipList)) { |
1360 |
+ REGION_INTERSECT(pScreen, src.reg, src.reg, &((WindowPtr)pSrc)->clipList); |
1361 |
} |
1362 |
+ |
1363 |
REGION_TRANSLATE(pScreen, src.reg, |
1364 |
dstx + pDst->x - srcx - pSrc->x, |
1365 |
dsty + pDst->y - srcy - pSrc->y); |
1366 |
|
1367 |
--- unix/xserver/hw/vnc/xf86vncModule.cc 2010-04-23 22:55:10.000000000 +0900 |
1368 |
+++ unix/xserver/hw/vnc/xf86vncModule.cc 2013-05-27 18:54:41.207573000 +0900 |
1369 |
@@ -25,6 +25,10 @@ |
1370 |
#include <rfb/Configuration.h> |
1371 |
#include <rfb/Logger_stdio.h> |
1372 |
#include <rfb/LogWriter.h> |
1373 |
+#include <rfb/ScreenSet.h> |
1374 |
+#include <rfb/screenTypes.h> |
1375 |
+ |
1376 |
+#include "xorg-version.h" |
1377 |
|
1378 |
extern "C" { |
1379 |
#define class c_class |
1380 |
@@ -33,6 +37,9 @@ |
1381 |
#define new c_new |
1382 |
#include "xf86.h" |
1383 |
#include "xf86Module.h" |
1384 |
+#ifdef RANDR |
1385 |
+#include "randrstr.h" |
1386 |
+#endif /* RANDR */ |
1387 |
#undef class |
1388 |
#undef private |
1389 |
#undef bool |
1390 |
@@ -89,7 +96,12 @@ |
1391 |
ScrnInfoPtr pScrn = xf86Screens[scr]; |
1392 |
|
1393 |
for (ParameterIterator i(Configuration::global()); i.param; i.next()) { |
1394 |
- char* val = xf86FindOptionValue(pScrn->options, i.param->getName()); |
1395 |
+ const char *val; |
1396 |
+#if XORG < 112 |
1397 |
+ val = xf86FindOptionValue(pScrn->options, i.param->getName()); |
1398 |
+#else |
1399 |
+ val = xf86FindOptionValue((XF86OptionPtr)pScrn->options, i.param->getName()); |
1400 |
+#endif |
1401 |
if (val) |
1402 |
i.param->setParam(val); |
1403 |
} |
1404 |
@@ -98,3 +110,13 @@ |
1405 |
vncExtensionInit(); |
1406 |
} |
1407 |
} |
1408 |
+ |
1409 |
+RRModePtr vncRandRModeGet(int width, int height) |
1410 |
+{ |
1411 |
+ return NULL; |
1412 |
+} |
1413 |
+ |
1414 |
+RROutputPtr vncRandROutputCreate(ScreenPtr pScreen) |
1415 |
+{ |
1416 |
+ return NULL; |
1417 |
+} |
1418 |
|
1419 |
--- unix/xserver/hw/vnc/xorg-version.h 2010-12-09 01:05:48.000000000 +0900 |
1420 |
+++ unix/xserver/hw/vnc/xorg-version.h 2013-05-27 18:54:41.193575000 +0900 |
1421 |
@@ -36,6 +36,10 @@ |
1422 |
#define XORG 19 |
1423 |
#elif XORG_VERSION_CURRENT < ((1 * 10000000) + (10 * 100000) + (99 * 1000)) |
1424 |
#define XORG 110 |
1425 |
+#elif XORG_VERSION_CURRENT < ((1 * 10000000) + (11 * 100000) + (99 * 1000)) |
1426 |
+#define XORG 111 |
1427 |
+#elif XORG_VERSION_CURRENT < ((1 * 10000000) + (12 * 100000) + (99 * 1000)) |
1428 |
+#define XORG 112 |
1429 |
#else |
1430 |
#error "X.Org newer than 1.10 is not supported" |
1431 |
#endif |
1432 |
|
1433 |
--- unix/xserver/hw/vnc/xvnc.cc 2012-03-10 05:34:29.000000000 +0900 |
1434 |
+++ unix/xserver/hw/vnc/xvnc.cc 2013-05-27 18:54:41.182576000 +0900 |
1435 |
@@ -96,7 +96,7 @@ |
1436 |
#define Xfree free |
1437 |
#endif |
1438 |
|
1439 |
-#define XVNCVERSION "TigerVNC 1.2.0" |
1440 |
+#define XVNCVERSION "TigerVNC 1.2.80" |
1441 |
#define XVNCCOPYRIGHT ("Copyright (C) 1999-2011 TigerVNC Team and many others (see README.txt)\n" \ |
1442 |
"See http://www.tigervnc.org for information on TigerVNC.\n") |
1443 |
|
1444 |
@@ -229,7 +229,11 @@ |
1445 |
} |
1446 |
#endif |
1447 |
|
1448 |
+#if XORG < 111 |
1449 |
void ddxGiveUp() |
1450 |
+#else |
1451 |
+void ddxGiveUp(enum ExitCode error) |
1452 |
+#endif |
1453 |
{ |
1454 |
int i; |
1455 |
|
1456 |
@@ -239,9 +243,17 @@ |
1457 |
} |
1458 |
|
1459 |
void |
1460 |
+#if XORG < 111 |
1461 |
AbortDDX() |
1462 |
+#else |
1463 |
+AbortDDX(enum ExitCode error) |
1464 |
+#endif |
1465 |
{ |
1466 |
+#if XORG < 111 |
1467 |
ddxGiveUp(); |
1468 |
+#else |
1469 |
+ ddxGiveUp(error); |
1470 |
+#endif |
1471 |
} |
1472 |
|
1473 |
#ifdef __DARWIN__ |
1474 |
@@ -686,8 +698,13 @@ |
1475 |
{ |
1476 |
if (pmap->mid != pmap->pScreen->defColormap) |
1477 |
{ |
1478 |
+#if XORG < 111 |
1479 |
curpmap = (ColormapPtr) LookupIDByType(pmap->pScreen->defColormap, |
1480 |
RT_COLORMAP); |
1481 |
+#else |
1482 |
+ dixLookupResourceByType((pointer *) &curpmap, pmap->pScreen->defColormap, |
1483 |
+ RT_COLORMAP, serverClient, DixUnknownAccess); |
1484 |
+#endif |
1485 |
(*pmap->pScreen->InstallColormap)(curpmap); |
1486 |
} |
1487 |
} |
1488 |
@@ -859,45 +876,8 @@ |
1489 |
|
1490 |
static Bool vncRandRGetInfo (ScreenPtr pScreen, Rotation *rotations) |
1491 |
{ |
1492 |
- Bool ret, gotCurrent = FALSE; |
1493 |
- int i; |
1494 |
- |
1495 |
- const int widths[] = { 1920, 1920, 1600, 1680, 1400, 1360, 1280, 1280, 1280, 1280, 1024, 800, 640 }; |
1496 |
- const int heights[] = { 1200, 1080, 1200, 1050, 1050, 768, 1024, 960, 800, 720, 768, 600, 480 }; |
1497 |
- |
1498 |
- for (i = 0;i < sizeof(widths)/sizeof(*widths);i++) { |
1499 |
- RRScreenSizePtr pSize; |
1500 |
- |
1501 |
- pSize = RRRegisterSize(pScreen, widths[i], heights[i], |
1502 |
- pScreen->mmWidth, pScreen->mmHeight); |
1503 |
- if (!pSize) |
1504 |
- return FALSE; |
1505 |
- |
1506 |
- ret = RRRegisterRate(pScreen, pSize, 60); |
1507 |
- if (!ret) |
1508 |
- return FALSE; |
1509 |
- |
1510 |
- if ((widths[i] == pScreen->width) && (heights[i] == pScreen->height)) { |
1511 |
- RRSetCurrentConfig(pScreen, RR_Rotate_0, 60, pSize); |
1512 |
- gotCurrent = TRUE; |
1513 |
- } |
1514 |
- } |
1515 |
- |
1516 |
- if (!gotCurrent) { |
1517 |
- RRScreenSizePtr pSize; |
1518 |
- |
1519 |
- pSize = RRRegisterSize(pScreen, pScreen->width, pScreen->height, |
1520 |
- pScreen->mmWidth, pScreen->mmHeight); |
1521 |
- if (!pSize) |
1522 |
- return FALSE; |
1523 |
- |
1524 |
- RRRegisterRate(pScreen, pSize, 60); |
1525 |
- |
1526 |
- RRSetCurrentConfig(pScreen, RR_Rotate_0, 60, pSize); |
1527 |
- } |
1528 |
- |
1529 |
- *rotations = RR_Rotate_0; |
1530 |
- |
1531 |
+ // We update all information right away, so there is nothing to |
1532 |
+ // do here. |
1533 |
return TRUE; |
1534 |
} |
1535 |
|
1536 |
@@ -1050,16 +1030,19 @@ |
1537 |
FlushAllOutput (); |
1538 |
} |
1539 |
|
1540 |
-static Bool vncRandRSetConfig (ScreenPtr pScreen, Rotation rotation, |
1541 |
- int rate, RRScreenSizePtr pSize) |
1542 |
+RRModePtr vncRandRModeGet(int width, int height); |
1543 |
+ |
1544 |
+static Bool vncRandRScreenSetSize(ScreenPtr pScreen, |
1545 |
+ CARD16 width, CARD16 height, |
1546 |
+ CARD32 mmWidth, CARD32 mmHeight) |
1547 |
{ |
1548 |
vfbScreenInfoPtr pvfb = &vfbScreens[pScreen->myNum]; |
1549 |
vfbFramebufferInfo fb; |
1550 |
+ rrScrPrivPtr rp = rrGetScrPriv(pScreen); |
1551 |
PixmapPtr rootPixmap = pScreen->GetScreenPixmap(pScreen); |
1552 |
void *pbits; |
1553 |
Bool ret; |
1554 |
int oldwidth, oldheight, oldmmWidth, oldmmHeight; |
1555 |
- int dpix, dpiy; |
1556 |
|
1557 |
/* Prevent updates while we fiddle */ |
1558 |
xf86SetRootClip(pScreen, FALSE); |
1559 |
@@ -1070,17 +1053,11 @@ |
1560 |
oldmmWidth = pScreen->mmWidth; |
1561 |
oldmmHeight = pScreen->mmHeight; |
1562 |
|
1563 |
- /* Compute the current DPI (for use later) */ |
1564 |
- dpix = (pScreen->width * 254 + pScreen->mmWidth * 5) / (pScreen->mmWidth * 10); |
1565 |
- dpiy = (pScreen->height * 254 + pScreen->mmHeight * 5) / (pScreen->mmHeight * 10); |
1566 |
- |
1567 |
/* Then set the new dimensions */ |
1568 |
- pScreen->width = pSize->width; |
1569 |
- pScreen->height = pSize->height; |
1570 |
- |
1571 |
- /* Try to keep the same DPI as we do not have a physical screen */ |
1572 |
- pScreen->mmWidth = (pScreen->width * 254 + dpix * 5) / (dpix * 10); |
1573 |
- pScreen->mmHeight = (pScreen->height * 254 + dpiy * 5) / (dpiy * 10); |
1574 |
+ pScreen->width = width; |
1575 |
+ pScreen->height = height; |
1576 |
+ pScreen->mmWidth = mmWidth; |
1577 |
+ pScreen->mmHeight = mmHeight; |
1578 |
|
1579 |
/* Allocate a new framebuffer */ |
1580 |
memset(&fb, 0, sizeof(vfbFramebufferInfo)); |
1581 |
@@ -1130,6 +1107,207 @@ |
1582 |
/* Restore ability to update screen, now with new dimensions */ |
1583 |
xf86SetRootClip(pScreen, TRUE); |
1584 |
|
1585 |
+ /* |
1586 |
+ * Let RandR know we changed something (it doesn't assume that |
1587 |
+ * TRUE means something changed for some reason...). |
1588 |
+ */ |
1589 |
+ RRScreenSizeNotify(pScreen); |
1590 |
+ |
1591 |
+ /* Crop all CRTCs to the new screen */ |
1592 |
+ for (int i = 0;i < rp->numCrtcs;i++) { |
1593 |
+ RRCrtcPtr crtc; |
1594 |
+ RRModePtr mode; |
1595 |
+ |
1596 |
+ crtc = rp->crtcs[i]; |
1597 |
+ |
1598 |
+ /* Disabled? */ |
1599 |
+ if (crtc->mode == NULL) |
1600 |
+ continue; |
1601 |
+ |
1602 |
+ /* Fully inside? */ |
1603 |
+ if ((crtc->x + crtc->mode->mode.width <= width) && |
1604 |
+ (crtc->y + crtc->mode->mode.height <= height)) |
1605 |
+ continue; |
1606 |
+ |
1607 |
+ /* Fully outside? */ |
1608 |
+ if ((crtc->x >= width) || (crtc->y >= height)) { |
1609 |
+ /* Disable it */ |
1610 |
+ ret = RRCrtcNotify(crtc, NULL, crtc->x, crtc->y, crtc->rotation, |
1611 |
+#if XORG >= 16 |
1612 |
+ NULL, |
1613 |
+#endif |
1614 |
+ crtc->numOutputs, crtc->outputs); |
1615 |
+ if (!ret) |
1616 |
+ ErrorF("Warning: Unable to disable CRTC that is outside of new screen dimensions"); |
1617 |
+ continue; |
1618 |
+ } |
1619 |
+ |
1620 |
+ /* Just needs to be resized */ |
1621 |
+ mode = vncRandRModeGet(width - crtc->x, height - crtc->y); |
1622 |
+ if (mode == NULL) { |
1623 |
+ ErrorF("Warning: Unable to create custom mode for %dx%d", |
1624 |
+ width - crtc->x, height - crtc->y); |
1625 |
+ continue; |
1626 |
+ } |
1627 |
+ |
1628 |
+ ret = RRCrtcNotify(crtc, mode, crtc->x, crtc->y, crtc->rotation, |
1629 |
+#if XORG >= 16 |
1630 |
+ NULL, |
1631 |
+#endif |
1632 |
+ crtc->numOutputs, crtc->outputs); |
1633 |
+ RRModeDestroy(mode); |
1634 |
+ if (!ret) |
1635 |
+ ErrorF("Warning: Unable to crop CRTC to new screen dimensions"); |
1636 |
+ } |
1637 |
+ |
1638 |
+ return TRUE; |
1639 |
+} |
1640 |
+ |
1641 |
+static Bool vncRandRCrtcSet(ScreenPtr pScreen, RRCrtcPtr crtc, RRModePtr mode, |
1642 |
+ int x, int y, Rotation rotation, int num_outputs, |
1643 |
+ RROutputPtr *outputs) |
1644 |
+{ |
1645 |
+ Bool ret; |
1646 |
+ |
1647 |
+ /* Let RandR know we approve, and let it update its internal state */ |
1648 |
+ ret = RRCrtcNotify(crtc, mode, x, y, rotation, |
1649 |
+#if XORG >= 16 |
1650 |
+ NULL, |
1651 |
+#endif |
1652 |
+ num_outputs, outputs); |
1653 |
+ if (!ret) |
1654 |
+ return FALSE; |
1655 |
+ |
1656 |
+ return TRUE; |
1657 |
+} |
1658 |
+ |
1659 |
+static Bool vncRandROutputValidateMode(ScreenPtr pScreen, |
1660 |
+ RROutputPtr output, RRModePtr mode) |
1661 |
+{ |
1662 |
+ /* We have no hardware so any mode works */ |
1663 |
+ return TRUE; |
1664 |
+} |
1665 |
+ |
1666 |
+static void vncRandRModeDestroy(ScreenPtr pScreen, RRModePtr mode) |
1667 |
+{ |
1668 |
+ /* We haven't allocated anything so nothing to destroy */ |
1669 |
+} |
1670 |
+ |
1671 |
+static const int vncRandRWidths[] = { 1920, 1920, 1600, 1680, 1400, 1360, 1280, 1280, 1280, 1280, 1024, 800, 640 }; |
1672 |
+static const int vncRandRHeights[] = { 1200, 1080, 1200, 1050, 1050, 768, 1024, 960, 800, 720, 768, 600, 480 }; |
1673 |
+ |
1674 |
+static int vncRandRIndex = 0; |
1675 |
+ |
1676 |
+/* This is a global symbol since XserverDesktop also uses it */ |
1677 |
+RRModePtr vncRandRModeGet(int width, int height) |
1678 |
+{ |
1679 |
+ xRRModeInfo modeInfo; |
1680 |
+ char name[100]; |
1681 |
+ RRModePtr mode; |
1682 |
+ |
1683 |
+ memset(&modeInfo, 0, sizeof(modeInfo)); |
1684 |
+ sprintf(name, "%dx%d", width, height); |
1685 |
+ |
1686 |
+ modeInfo.width = width; |
1687 |
+ modeInfo.height = height; |
1688 |
+ modeInfo.hTotal = width; |
1689 |
+ modeInfo.vTotal = height; |
1690 |
+ modeInfo.dotClock = ((CARD32)width * (CARD32)height * 60); |
1691 |
+ modeInfo.nameLength = strlen(name); |
1692 |
+ mode = RRModeGet(&modeInfo, name); |
1693 |
+ if (mode == NULL) |
1694 |
+ return NULL; |
1695 |
+ |
1696 |
+ return mode; |
1697 |
+} |
1698 |
+ |
1699 |
+static RRCrtcPtr vncRandRCrtcCreate(ScreenPtr pScreen) |
1700 |
+{ |
1701 |
+ RRCrtcPtr crtc; |
1702 |
+ RROutputPtr output; |
1703 |
+ RRModePtr mode; |
1704 |
+ char name[100]; |
1705 |
+ |
1706 |
+ /* First we create the CRTC... */ |
1707 |
+ crtc = RRCrtcCreate(pScreen, NULL); |
1708 |
+ |
1709 |
+ /* We don't actually support gamma, but xrandr complains when it is missing */ |
1710 |
+ RRCrtcGammaSetSize (crtc, 256); |
1711 |
+ |
1712 |
+ /* Then we create a dummy output for it... */ |
1713 |
+ sprintf(name, "VNC-%d", vncRandRIndex); |
1714 |
+ vncRandRIndex++; |
1715 |
+ |
1716 |
+ output = RROutputCreate(pScreen, name, strlen(name), NULL); |
1717 |
+ |
1718 |
+ RROutputSetCrtcs(output, &crtc, 1); |
1719 |
+ RROutputSetConnection(output, RR_Connected); |
1720 |
+ |
1721 |
+ /* Make sure the CRTC has this output set */ |
1722 |
+ RRCrtcNotify(crtc, NULL, 0, 0, RR_Rotate_0, |
1723 |
+#if XORG >= 16 |
1724 |
+ NULL, |
1725 |
+#endif |
1726 |
+ 1, &output); |
1727 |
+ |
1728 |
+ /* Populate a list of default modes */ |
1729 |
+ RRModePtr modes[sizeof(vncRandRWidths)/sizeof(*vncRandRWidths)]; |
1730 |
+ int num_modes; |
1731 |
+ |
1732 |
+ num_modes = 0; |
1733 |
+ for (int i = 0;i < sizeof(vncRandRWidths)/sizeof(*vncRandRWidths);i++) { |
1734 |
+ mode = vncRandRModeGet(vncRandRWidths[i], vncRandRHeights[i]); |
1735 |
+ if (mode != NULL) { |
1736 |
+ modes[num_modes] = mode; |
1737 |
+ num_modes++; |
1738 |
+ } |
1739 |
+ } |
1740 |
+ |
1741 |
+ RROutputSetModes(output, modes, num_modes, 0); |
1742 |
+ |
1743 |
+ return crtc; |
1744 |
+} |
1745 |
+ |
1746 |
+/* Used from XserverDesktop when it needs more outputs... */ |
1747 |
+RROutputPtr vncRandROutputCreate(ScreenPtr pScreen) |
1748 |
+{ |
1749 |
+ RRCrtcPtr crtc; |
1750 |
+ |
1751 |
+ crtc = vncRandRCrtcCreate(pScreen); |
1752 |
+ if (crtc == NULL) |
1753 |
+ return NULL; |
1754 |
+ |
1755 |
+ return crtc->outputs[0]; |
1756 |
+} |
1757 |
+ |
1758 |
+static Bool vncRandRInit(ScreenPtr pScreen) |
1759 |
+{ |
1760 |
+ RRCrtcPtr crtc; |
1761 |
+ RRModePtr mode; |
1762 |
+ |
1763 |
+ if (!RRInit()) |
1764 |
+ return FALSE; |
1765 |
+ |
1766 |
+ /* These are completely arbitrary */ |
1767 |
+ RRScreenSetSizeRange(pScreen, 32, 32, 32768, 32768); |
1768 |
+ |
1769 |
+ /* |
1770 |
+ * Start with a single CRTC with a single output. More will be |
1771 |
+ * allocated as needed... |
1772 |
+ */ |
1773 |
+ crtc = vncRandRCrtcCreate(pScreen); |
1774 |
+ |
1775 |
+ /* Make sure the current screen size is the active mode */ |
1776 |
+ mode = vncRandRModeGet(pScreen->width, pScreen->height); |
1777 |
+ if (mode == NULL) |
1778 |
+ return FALSE; |
1779 |
+ |
1780 |
+ RRCrtcNotify(crtc, mode, 0, 0, RR_Rotate_0, |
1781 |
+#if XORG >= 16 |
1782 |
+ NULL, |
1783 |
+#endif |
1784 |
+ crtc->numOutputs, crtc->outputs); |
1785 |
+ |
1786 |
return TRUE; |
1787 |
} |
1788 |
|
1789 |
@@ -1291,8 +1469,16 @@ |
1790 |
if (!ret) return FALSE; |
1791 |
|
1792 |
rp = rrGetScrPriv(pScreen); |
1793 |
+ |
1794 |
rp->rrGetInfo = vncRandRGetInfo; |
1795 |
- rp->rrSetConfig = vncRandRSetConfig; |
1796 |
+ rp->rrSetConfig = NULL; |
1797 |
+ rp->rrScreenSetSize = vncRandRScreenSetSize; |
1798 |
+ rp->rrCrtcSet = vncRandRCrtcSet; |
1799 |
+ rp->rrOutputValidateMode = vncRandROutputValidateMode; |
1800 |
+ rp->rrModeDestroy = vncRandRModeDestroy; |
1801 |
+ |
1802 |
+ ret = vncRandRInit(pScreen); |
1803 |
+ if (!ret) return FALSE; |
1804 |
#endif |
1805 |
|
1806 |
|