X-Git-Url: http://git.tdb.fi/?a=blobdiff_plain;f=source%2Fmain.c;h=15486e61c83f3dfef06c88816d1f4db2ade4985b;hb=ce6b94be7d334af14fac9f997ab66ff9a9aeecc5;hp=35328543cbbb4cfc80329c8a531870ab6690aaa0;hpb=42750151d3f9fed56dc333ec7949c9a8005570b0;p=geometrycompositor.git diff --git a/source/main.c b/source/main.c index 3532854..15486e6 100644 --- a/source/main.c +++ b/source/main.c @@ -16,6 +16,7 @@ typedef struct CompositedWindow int y; unsigned width; unsigned height; + unsigned border; int map_state; Damage damage; Pixmap pixmap; @@ -42,6 +43,7 @@ typedef struct CompositedScreen CompositedWindow *windows; unsigned nwindows; unsigned windows_capacity; + int dirty; } CompositedScreen; typedef struct Compositor @@ -53,6 +55,7 @@ typedef struct Compositor PFNGLXCREATECONTEXTATTRIBSARBPROC glXCreateContextAttribsARB; PFNGLXBINDTEXIMAGEEXTPROC glXBindTexImageEXT; PFNGLXRELEASETEXIMAGEEXTPROC glXReleaseTexImageEXT; + int dirty; } Compositor; static const char *vshader = @@ -84,6 +87,13 @@ static const float vertices[] = 1.0f, 0.0f }; +int x_error_handler(Display *display, XErrorEvent *event) +{ + printf("Ignoring X error %d on resource %lx\n", event->error_code, event->resourceid); + (void)display; + return 0; +} + int with_error(const char *message) { fprintf(stderr, "%s\n", message); @@ -255,7 +265,11 @@ void add_window(Compositor *compositor, CompositedScreen *screen, Window w) if(w==screen->root || w==screen->overlay) return; - XGetWindowAttributes(compositor->display, w, &win_attr); + if(!XGetWindowAttributes(compositor->display, w, &win_attr)) + { + printf("XGetWindowAttributes failed; probably the window was already destroyed\n"); + return; + } if(win_attr.class==InputOnly) return; @@ -275,6 +289,7 @@ void add_window(Compositor *compositor, CompositedScreen *screen, Window w) window->y = win_attr.y; window->width = win_attr.width; window->height = win_attr.height; + window->border = win_attr.border_width; window->map_state = win_attr.map_state; window->damage = XDamageCreate(compositor->display, window->window, XDamageReportNonEmpty); @@ -303,6 +318,18 @@ CompositedScreen *find_screen_by_root(Compositor *compositor, Window root) return NULL; } +CompositedScreen *find_screen_by_window(Compositor *compositor, Window w) +{ + unsigned i, j; + + for(i=0; inscreens; ++i) + for(j=0; jscreens[i].nwindows; ++j) + if(compositor->screens[i].windows[j].window==w) + return &compositor->screens[i]; + + return NULL; +} + int initialize_screen(Compositor *compositor, unsigned number) { CompositedScreen *screen; @@ -348,6 +375,8 @@ int initialize_screen(Compositor *compositor, unsigned number) XFree(children); + screen->dirty = 1; + return 1; } @@ -363,6 +392,8 @@ int initialize_compositor(Compositor *compositor) if(!compositor->display) return with_error("Could not open X display"); + XSetErrorHandler(&x_error_handler); + if(!XCompositeQueryExtension(compositor->display, &event_base, &error_base)) return with_error("XComposite is required but was not found"); else if(!XCompositeQueryVersion(compositor->display, &major_ver, &minor_ver)) @@ -401,6 +432,8 @@ int initialize_compositor(Compositor *compositor) if(!initialize_screen(compositor, i)) return 0; + compositor->dirty = 1; + return 1; } @@ -456,10 +489,27 @@ void process_unmap_event(Compositor *compositor, XUnmapEvent *event) window->map_state = IsUnviewable; } -void process_event(Compositor *compositor) +void process_damage_event(Compositor *compositor, XDamageNotifyEvent *event) +{ + CompositedScreen *screen = find_screen_by_window(compositor, event->drawable); + if(!screen) + return; + + screen->dirty = 1; + compositor->dirty = 1; +} + +int process_event(Compositor *compositor) { XEvent event; - XNextEvent(compositor->display, &event); + if(compositor->dirty) + { + if(!XCheckMaskEvent(compositor->display, -1, &event)) + return 0; + } + else + XNextEvent(compositor->display, &event); + switch(event.type) { case CreateNotify: @@ -472,9 +522,13 @@ void process_event(Compositor *compositor) process_unmap_event(compositor, &event.xunmap); break; default: - if(event.type!=compositor->damage_event+XDamageNotify) + if(event.type==compositor->damage_event+XDamageNotify) + process_damage_event(compositor, (XDamageNotifyEvent *)&event); + else printf("Event %d\n", event.type); } + + return 1; } void refresh_screens(Compositor *compositor) @@ -491,8 +545,6 @@ void refresh_screens(Compositor *compositor) glUseProgram(screen->program); glBindVertexArray(screen->vertex_array); - XGrabServer(compositor->display); - glXWaitX(); for(j=0; jnwindows; ++j) { CompositedWindow *window = &screen->windows[j]; @@ -500,15 +552,18 @@ void refresh_screens(Compositor *compositor) { glBindTexture(GL_TEXTURE_2D, window->texture); compositor->glXBindTexImageEXT(compositor->display, window->glx_pixmap, GLX_FRONT_LEFT_EXT, NULL); - glUniform4f(screen->geometry_loc, (float)window->x/screen->width, (float)(screen->height-window->y-window->height)/screen->height, (float)window->width/screen->width, (float)window->height/screen->height); + glUniform4f(screen->geometry_loc, (float)window->x/screen->width, (float)(screen->height-window->y-window->height)/screen->height, (float)(window->width+2*window->border)/screen->width, (float)(window->height+2*window->border)/screen->height); glDrawArrays(GL_TRIANGLE_STRIP, 0, 4); compositor->glXReleaseTexImageEXT(compositor->display, window->glx_pixmap, GLX_FRONT_LEFT_EXT); XDamageSubtract(compositor->display, window->damage, None, None); } } - XUngrabServer(compositor->display); glXSwapBuffers(compositor->display, screen->glx_window); + + screen->dirty = 0; } + + compositor->dirty = 0; } int main() @@ -520,8 +575,8 @@ int main() while(1) { - process_event(&compositor); - refresh_screens(&compositor); + if(!process_event(&compositor)) + refresh_screens(&compositor); } shutdown_compositor(&compositor);