3 #include <X11/extensions/Xrandr.h>
5 #include <msp/io/print.h>
6 #include <msp/strings/format.h>
7 #include <msp/strings/lexicalcast.h>
9 #include "display_private.h"
15 bool error_flag = false;
16 std::string error_msg;
18 int x_error_handler(Display *display, XErrorEvent *event)
21 XGetErrorText(display, event->error_code, err, sizeof(err));
23 string request_code = Msp::lexical_cast<string, int>(event->request_code);
25 XGetErrorDatabaseText(display, "XRequest", request_code.c_str(), request_code.c_str(), req, sizeof(req));
27 string msg = Msp::format("Request %s failed with %s [%08X]", req, err, event->resourceid);
29 Msp::IO::print(Msp::IO::cerr, "Discarding error: %s\n", msg);
32 Msp::IO::print(Msp::IO::cerr, "%s\n", msg);
41 inline Msp::Graphics::VideoRotation rotation_from_sys(Rotation r)
45 case RR_Rotate_90: return Msp::Graphics::ROTATE_RIGHT;
46 case RR_Rotate_180: return Msp::Graphics::ROTATE_INVERTED;
47 case RR_Rotate_270: return Msp::Graphics::ROTATE_LEFT;
48 default: return Msp::Graphics::ROTATE_NORMAL;
52 inline Rotation rotation_to_sys(Msp::Graphics::VideoRotation r)
56 case Msp::Graphics::ROTATE_RIGHT: return RR_Rotate_90;
57 case Msp::Graphics::ROTATE_INVERTED: return RR_Rotate_180;
58 case Msp::Graphics::ROTATE_LEFT: return RR_Rotate_270;
59 default: return RR_Rotate_0;
63 inline bool monitor_x_compare(const Msp::Graphics::Monitor &m1, const Msp::Graphics::Monitor &m2)
65 if(m1.desktop_mode && !m2.desktop_mode)
71 inline unsigned mode_width(const Msp::Graphics::VideoMode &m, Msp::Graphics::VideoRotation r)
73 if(r==Msp::Graphics::ROTATE_RIGHT || r==Msp::Graphics::ROTATE_LEFT)
85 Display::Display(const string &disp_name):
90 priv->display = XOpenDisplay(0);
92 priv->display = XOpenDisplay(disp_name.c_str());
94 throw runtime_error("XOpenDisplay");
96 XSetErrorHandler(x_error_handler);
98 priv->root_window = DefaultRootWindow(priv->display);
100 err_dialog = new ErrorDialog(this);
105 if(XRRQueryExtension(priv->display, &event_base, &error_base))
108 XRRQueryVersion(priv->display, &major, &minor);
109 if(major>1 || (major==1 && minor>=2))
111 XRRScreenResources *res = XRRGetScreenResources(priv->display, priv->root_window);
112 RROutput primary = XRRGetOutputPrimary(priv->display, priv->root_window);
114 map<RRMode, XRRModeInfo *> modes_by_id;
115 for(int i=0; i<res->nmode; ++i)
116 modes_by_id[res->modes[i].id] = &res->modes[i];
118 for(int i=0; i<res->noutput; ++i)
120 XRROutputInfo *output = XRRGetOutputInfo(priv->display, res, res->outputs[i]);
121 XRRCrtcInfo *crtc = (output->crtc ? XRRGetCrtcInfo(priv->display, res, output->crtc) : 0);
123 monitors.push_back(Monitor());
124 Monitor &monitor = monitors.back();
125 monitor.index = monitors.size()-1;
126 monitor.name.assign(output->name, output->nameLen);
127 priv->monitors.push_back(res->outputs[i]);
131 monitor.desktop_rotation = rotation_from_sys(crtc->rotation);
132 monitor.current_rotation = monitor.desktop_rotation;
137 if(res->outputs[i]==primary)
138 primary_monitor = &monitor;
140 for(int j=0; j<output->nmode; ++j)
142 map<RRMode, XRRModeInfo *>::iterator k = modes_by_id.find(output->modes[j]);
143 if(k==modes_by_id.end())
146 XRRModeInfo *info = k->second;
148 VideoMode mode(info->width, info->height);
149 mode.index = modes.size();
150 mode.monitor = &monitor;
151 mode.rate = static_cast<float>(info->dotClock)/(info->hTotal*info->vTotal);
152 if(find_mode(mode, 0.01f))
155 modes.push_back(mode);
156 priv->modes.push_back(info->id);
157 monitor.video_modes.push_back(&modes.back());
159 if(crtc && info->id==crtc->mode)
161 monitor.desktop_mode = &modes.back();
162 monitor.current_mode = monitor.desktop_mode;
166 XRRFreeOutputInfo(output);
168 XRRFreeCrtcInfo(crtc);
171 XRRFreeScreenResources(res);
173 monitors.sort(monitor_x_compare);
174 Monitor *prev_enabled = 0;
175 for(list<Monitor>::iterator i=monitors.begin(); i!=monitors.end(); ++i)
178 i->next_left = prev_enabled;
180 prev_enabled->next_right = &*i;
184 if(!primary_monitor || !primary_monitor->desktop_mode)
186 // XRandR didn't give a sensible primary monitor. Try to guess one.
187 unsigned largest = 0;
188 for(list<Monitor>::iterator i=monitors.begin(); i!=monitors.end(); ++i)
191 unsigned size = i->desktop_mode->width*i->desktop_mode->height;
195 primary_monitor = &*i;
206 XCloseDisplay(priv->display);
211 void Display::set_mode(const VideoMode &requested_mode, bool exclusive)
214 const VideoMode *mode = find_mode(requested_mode);
216 throw unsupported_video_mode(requested_mode);
218 VideoRotation requested_rotation = requested_mode.rotation;
219 if(requested_rotation==ROTATE_ANY)
220 requested_rotation = mode->monitor->desktop_rotation;
222 XRRScreenResources *res = XRRGetScreenResources(priv->display, priv->root_window);
223 RROutput output = priv->monitors[mode->monitor->index];
224 XRROutputInfo *output_info = XRRGetOutputInfo(priv->display, res, output);
226 // Check if the output already has a CRTC and find a free one if it doesn't
227 RRCrtc crtc = output_info->crtc;
228 XRRCrtcInfo *crtc_info = 0;
230 crtc_info = XRRGetCrtcInfo(priv->display, res, crtc);
233 for(int i=0; i<res->ncrtc; ++i)
235 crtc_info = XRRGetCrtcInfo(priv->display, res, res->crtcs[i]);
236 if(!crtc_info->noutput)
238 crtc = res->crtcs[i];
241 XRRFreeCrtcInfo(crtc_info);
246 XRRFreeOutputInfo(output_info);
247 throw unsupported_video_mode(requested_mode);
256 // Disable other outputs for exclusive mode
257 for(list<Monitor>::iterator i=monitors.begin(); i!=monitors.end(); ++i)
258 if(&*i!=mode->monitor)
260 XRROutputInfo *o = XRRGetOutputInfo(priv->display, res, priv->monitors[i->index]);
262 XRRSetCrtcConfig(priv->display, res, o->crtc, CurrentTime, 0, 0, 0, RR_Rotate_0, 0, 0);
263 XRRFreeOutputInfo(o);
266 i->current_rotation = ROTATE_NORMAL;
273 const Monitor *left = mode->monitor->next_left;
274 while(left && !left->current_mode)
275 left = left->next_left;
279 x = left->x+mode_width(*left->current_mode, left->current_rotation);
284 XRRSetCrtcConfig(priv->display, res, crtc, CurrentTime, x, y, priv->modes[mode->index], rotation_to_sys(requested_rotation), &output, 1);
286 list<Monitor>::iterator i;
287 for(i=monitors.begin(); i!=monitors.end(); ++i)
288 if(&*i==mode->monitor)
290 i->current_mode = mode;
291 i->current_rotation = requested_rotation;
295 x += mode_width(*mode, requested_rotation);
300 for(; i!=monitors.end(); ++i)
303 XRROutputInfo *o = XRRGetOutputInfo(priv->display, res, priv->monitors[i->index]);
304 XRRSetCrtcConfig(priv->display, res, o->crtc, CurrentTime, x, y, priv->modes[i->current_mode->index], rotation_to_sys(i->current_rotation), &priv->monitors[i->index], 1);
307 panning.timestamp = CurrentTime;
310 panning.width = i->current_mode->width;
311 panning.height = i->current_mode->height;
312 panning.track_left = panning.left;
313 panning.track_top = panning.top;
314 panning.track_width = panning.width;
315 panning.track_height = panning.height;
316 panning.border_left = 0;
317 panning.border_top = 0;
318 panning.border_right = 0;
319 panning.border_bottom = 0;
320 XRRSetPanning(priv->display, res, o->crtc, &panning);
322 XRRFreeOutputInfo(o);
327 x += mode_width(*i->current_mode, i->current_rotation);
330 XRRFreeOutputInfo(output_info);
331 XRRFreeCrtcInfo(crtc_info);
332 XRRFreeScreenResources(res);
334 (void)requested_mode;
336 throw runtime_error("no xrandr support");
340 bool Display::process_events()
342 int pending = XPending(priv->display);
349 XNextEvent(priv->display, &event.xevent);
353 map<WindowHandle, Window *>::iterator j = priv->windows.find(event.xevent.xany.window);
354 if(j!=priv->windows.end())
356 /* Filter keyboard autorepeat. If this packet is a KeyRelease and
357 the next one is a KeyPress with the exact same parameters, they
358 indicate autorepeat and must be dropped. */
359 if(event.xevent.type==KeyRelease && !j->second->get_keyboard_autorepeat() && pending>0)
361 XKeyEvent &kev = event.xevent.xkey;
363 XPeekEvent(priv->display, &ev2);
364 if(ev2.type==KeyPress)
366 XKeyEvent &kev2 = ev2.xkey;
367 if(kev2.window==kev.window && kev2.time==kev.time && kev2.keycode==kev.keycode)
369 XNextEvent(priv->display, &ev2);
376 j->second->event(event);
383 void Display::check_error()
388 throw runtime_error(error_msg);
393 } // namespace Graphics