/* * Error handling mechanism which permits us to ignore specific X11 * errors from particular requests. We maintain a list of upcoming * potential error events that we want to not treat as fatal errors. */ #include #include #include #include #include #include "putty.h" #ifndef NOT_X_WINDOWS #include #include #include #include "x11misc.h" static int (*orig_x11_error_handler)(Display *thisdisp, XErrorEvent *err); struct x11_err_to_ignore { Display *display; unsigned char error_code; unsigned long serial; }; static struct x11_err_to_ignore *errs; static size_t nerrs, errsize; static int x11_error_handler(Display *thisdisp, XErrorEvent *err) { for (size_t i = 0; i < nerrs; i++) { if (thisdisp == errs[i].display && err->serial == errs[i].serial && err->error_code == errs[i].error_code) { /* Ok, this is an error we're happy to ignore */ return 0; } } return (*orig_x11_error_handler)(thisdisp, err); } void x11_ignore_error(Display *disp, unsigned char errcode) { /* * Install our error handler, if we haven't already. */ if (!orig_x11_error_handler) orig_x11_error_handler = XSetErrorHandler(x11_error_handler); /* * This is as good a moment as any to winnow the ignore list based * on requests we know to have been processed. */ { unsigned long last = LastKnownRequestProcessed(disp); size_t i, j; for (i = j = 0; i < nerrs; i++) { if (errs[i].display == disp && errs[i].serial <= last) continue; errs[j++] = errs[i]; } nerrs = j; } sgrowarray(errs, errsize, nerrs); errs[nerrs].display = disp; errs[nerrs].error_code = errcode; errs[nerrs].serial = NextRequest(disp); nerrs++; } #else /* NOT_X_WINDOWS */ /* * Include _something_ in this file to prevent an annoying compiler * warning, and to avoid having to condition out this file in * CMakeLists. It's in a library, so this variable shouldn't end up in * any actual program, because nothing will refer to it. */ const int x11_ignore_error_dummy_variable = 0; #endif /* NOT_X_WINDOWS */