Exemple #1
0
static void
GGI_helper_x_dbe_finish(struct ggi_helper *helper)
{
#if 0
	ggi_x_priv *priv = GGIX_PRIV(helper->visual);
	XdbeBackBuffer backBuffer;

	/* Deallocate back buffer */
	if(!XdbeDeallocateBackBufferName(priv->disp, priv->drawable)) {
		DPRINT_LIBS("X: DOUBLE-BUFFER: Unable to deallocate back buffer.\n");
		return GGI_EFATAL;
	}
#endif
}
Exemple #2
0
void x11_cleanup(XRESOURCES* xres, CFG* config){
	if(!(xres->display)){
		return;
	}

	XftColorFree(xres->display, DefaultVisual(xres->display, xres->screen), DefaultColormap(xres->display, xres->screen), &(xres->text_color));
	XftColorFree(xres->display, DefaultVisual(xres->display, xres->screen), DefaultColormap(xres->display, xres->screen), &(xres->bg_color));
	XftColorFree(xres->display, DefaultVisual(xres->display, xres->screen), DefaultColormap(xres->display, xres->screen), &(xres->debug_color));
	if(xres->drawable){
		XftDrawDestroy(xres->drawable);
	}
	if(config->double_buffer){
		XdbeDeallocateBackBufferName(xres->display, xres->back_buffer);
	}
	XCloseDisplay(xres->display);
	xfd_free(&(xres->xfds));
}
Exemple #3
0
bool
_pl_y_end_page (S___(Plotter *_plotter))
{
    plColorRecord *cptr;
    plXFontRecord *fptr;
    Pixmap bg_pixmap = (Pixmap)0;
    int window_width, window_height;
    pid_t forkval;

    /* compute rectangle size; note flipped-y convention */
    window_width = (_plotter->data->imax - _plotter->data->imin) + 1;
    window_height = (_plotter->data->jmin - _plotter->data->jmax) + 1;

    /* if either sort of server-supported double buffering is being used,
       create background pixmap for Label widget (it doesn't yet have one) */
    if (_plotter->x_double_buffering == X_DBL_BUF_MBX
            || _plotter->x_double_buffering == X_DBL_BUF_DBE)
    {
        int screen;		/* screen number */
        Screen *screen_struct;	/* screen structure */

        screen = DefaultScreen (_plotter->x_dpy);
        screen_struct = ScreenOfDisplay (_plotter->x_dpy, screen);
        bg_pixmap = XCreatePixmap(_plotter->x_dpy,
                                  _plotter->x_drawable2,
                                  (unsigned int)window_width,
                                  (unsigned int)window_height,
                                  (unsigned int)PlanesOfScreen(screen_struct));

        /* copy from off-screen graphics buffer to pixmap */
        XCopyArea (_plotter->x_dpy, _plotter->x_drawable3, bg_pixmap,
                   _plotter->drawstate->x_gc_bg,
                   0, 0,
                   (unsigned int)window_width, (unsigned int)window_height,
                   0, 0);

        /* pixmap is installed below as background pixmap for Label widget */
    }

    /* If double buffering, must make final frame of graphics visible, by
       copying it from our off-screen graphics buffer `x_drawable3' to window.
       There are several types of double buffering: the two server-supported
       types, and the `by hand' type. */

#ifdef HAVE_X11_EXTENSIONS_XDBE_H
#ifdef HAVE_DBE_SUPPORT
    if (_plotter->x_double_buffering == X_DBL_BUF_DBE)
        /* we're using the X double buffering extension; off-screen graphics
           buffer `x_drawable3' is a back buffer */
    {
        XdbeSwapInfo info;

        /* make final frame of graphics visible by interchanging front and
           back buffers one last time */
        info.swap_window = _plotter->x_drawable2;
        info.swap_action = XdbeUndefined;
        XdbeSwapBuffers (_plotter->x_dpy, &info, 1);

        /* free the back buffer */
        XdbeDeallocateBackBufferName (_plotter->x_dpy, _plotter->x_drawable3);
    }
#endif /* HAVE_DBE_SUPPORT */
#endif /* HAVE_X11_EXTENSIONS_XDBE_H */

#ifdef HAVE_X11_EXTENSIONS_MULTIBUF_H
#ifdef HAVE_MBX_SUPPORT
    if (_plotter->x_double_buffering == X_DBL_BUF_MBX)
        /* we're using the X multibuffering extension; off-screen graphics
           buffer `x_drawable3' is a non-displayed multibuffer */
    {
        /* make final frame of graphics visible by making the multibuffer
        into which we're currently drawing the on-screen multibuffer */
        XmbufDisplayBuffers (_plotter->x_dpy, 1, &(_plotter->x_drawable3), 0, 0);
    }
#endif /* HAVE_MBX_SUPPORT */
#endif /* HAVE_X11_EXTENSIONS_MULTIBUF_H */

    /* if either sort of server-supported double buffering is being used,
       install the above-created pixmap as background pixmap for the Label
       widget to use, once the window has been spun off */
    if (_plotter->x_double_buffering == X_DBL_BUF_MBX
            || _plotter->x_double_buffering == X_DBL_BUF_DBE)
    {
        Arg wargs[2];		/* werewolves */

        /* install pixmap as Label widget's background pixmap */
#ifdef USE_MOTIF
        XtSetArg (wargs[0], XmNlabelPixmap, (Pixmap)bg_pixmap);
        XtSetArg (wargs[1], XmNlabelType, XmPIXMAP);
        XtSetValues (_plotter->y_canvas, wargs, (Cardinal)2);
#else
        XtSetArg (wargs[0], XtNbitmap, (Pixmap)bg_pixmap);
        XtSetValues (_plotter->y_canvas, wargs, (Cardinal)1);
#endif
    }

    if (_plotter->x_double_buffering == X_DBL_BUF_BY_HAND)
        /* we're double buffering _manually_, rather than using either X11
           protocol extension, so our off-screen graphics buffer `x_drawable3' is
           an ordinary pixmap */
    {
        /* make final frame of graphics visible by copying from pixmap to
               window */
        XCopyArea (_plotter->x_dpy, _plotter->x_drawable3, _plotter->x_drawable2,
                   _plotter->drawstate->x_gc_bg,
                   0, 0,
                   (unsigned int)window_width, (unsigned int)window_height,
                   0, 0);
    }

    /* Finally: if we're not double buffering at all, we copy our off-screen
       graphics buffer to the window.  The off-screen graphics buffer is just
       the Label widget's background pixmap, `x_drawable1'. */
    if (_plotter->x_double_buffering == X_DBL_BUF_NONE)
        XCopyArea (_plotter->x_dpy, _plotter->x_drawable1, _plotter->x_drawable2,
                   _plotter->drawstate->x_gc_bg,
                   0, 0,
                   (unsigned int)window_width, (unsigned int)window_height,
                   0, 0);

    /* following two deallocations (of font records and color cell records)
       arrange things so that when drawing the next page of graphics, which
       will require another connection to the X server, the Plotter will
       start with a clean slate */

    /* Free font records from Plotter's cache list.  This involves
       deallocating the font name and also the XFontStruct contained in each
       record, if non-NULL.  (NULL indicates that the font could not be
       retrieved.)  */
    fptr = _plotter->x_fontlist;
    _plotter->x_fontlist = NULL;
    while (fptr)
    {
        plXFontRecord *fptrnext;

        fptrnext = fptr->next;
        free (fptr->x_font_name);
        if (fptr->x_font_struct)
            XFreeFont (_plotter->x_dpy, fptr->x_font_struct);
        free (fptr);
        fptr = fptrnext;
    }

    /* Free cached color cells from Plotter's cache list.  Do _not_ ask the
       server to deallocate the cells themselves, because the child process
       will need them; just free local storage. */
    cptr = _plotter->x_colorlist;
    _plotter->x_colorlist = NULL;
    while (cptr)
    {
        plColorRecord *cptrnext;

        cptrnext = cptr->next;
        free (cptr);
        cptr = cptrnext;
    }

    /* A bit of last-minute cleanup (could be done elsewhere): call waitpid()
       to reclaim resources used by zombie child processes resulting from
       previous closepl()'s, if any.  If this isn't done, the controlling
       process of any previously popped-up window won't fully exit (e.g. when
       `q' is typed in the window): it'll remain in the process table as a
       zombie until the parent process executes. */
#ifdef HAVE_WAITPID
#ifdef HAVE_SYS_WAIT_H
#ifdef WNOHANG
    {
        int i;

        /* iterate over all previously forked-off children (should really keep
           track of which have exited, since once a child has exited, invoking
           waitpid() on it is pointless) */
        for (i = 0; i < _plotter->y_num_pids; i++)
            waitpid (_plotter->y_pids[i], (int *)NULL, WNOHANG);
    }
#endif
#endif
#endif

    /* maybe flush X output buffer and handle X events (a no-op for
       XDrawablePlotters, which is overridden for XPlotters) */
    _maybe_handle_x_events (S___(_plotter));

    /* flush out the X output buffer; wait till all requests have been
       received and processed by server (see x_flushpl.c) */
    _pl_x_flush_output (S___(_plotter));

    /* flush output streams for all Plotters before forking */
    _pl_g_flush_plotter_outstreams (S___(_plotter));

    /* DO IT, MAN! */
    forkval = fork ();
    if ((int)forkval > 0		/* fork succeeded, and we're the parent */
            || (int)forkval < 0)	/* fork failed */
    {
        bool retval = true;

        if ((int)forkval < 0)
            _plotter->error (R___(_plotter) "the process could not be forked");

        /* Close connection to X display associated with window that the
        child process should manage, i.e. with the last openpl() invoked
         on this Plotter. */
        if (close (ConnectionNumber (_plotter->x_dpy)) < 0
                && errno != EINTR)
            /* emphatically shouldn't happen */
        {
            _plotter->error (R___(_plotter) "the connection to the X display could not be closed");
            retval = false;
        }

        if ((int)forkval > 0)
            /* there's a child process, so save its pid */
        {
            if (_plotter->y_num_pids == 0)
                _plotter->y_pids = (pid_t *)_pl_xmalloc (sizeof (pid_t));
            else
                _plotter->y_pids =
                    (pid_t *)_pl_xrealloc (_plotter->y_pids,
                                           ((_plotter->y_num_pids + 1)
                                            * sizeof (pid_t)));
            _plotter->y_pids[_plotter->y_num_pids] = forkval;
            _plotter->y_num_pids++;
        }

        /* do teardown of X-specific elements of the first drawing state on
        the drawing state stack */
        _pl_x_delete_gcs_from_first_drawing_state (S___(_plotter));

        return retval;
    }

    else		/* forkval = 0; fork succeeded, and we're the child */
    {
        bool need_redisplay = false;
        int i;

        /* Alter canvas widget's translation table, so that exit will occur
        when `q' is typed (or mouse is clicked).  See y_openpl.c. */
        _pl_y_set_data_for_quitting (S___(_plotter));

        /* Close all connections to X display other than our own, i.e., close
        all connections that other XPlotters may have been using.  No need
         to lock the global variables _xplotters and _xplotters_len; since
         we've forked and we're the child process, we're the only thread
         left. :-)

         We'll never be accessing those variables again (the only way we
         could would be if we were to call _maybe_handle_x_events(), and
         we aren't going to do that).  So we don't need to worry that they
         may actually be locked.  I.e. there was no need for us to register
         a handler to unlock them immediately after forking, by invoking
         pthread_atfork().  Which is why we didn't do that. */

        for (i = 0; i < _xplotters_len; i++)
            if (_xplotters[i] != NULL
                    && _xplotters[i] != _plotter
                    && _xplotters[i]->data->opened
                    && _xplotters[i]->data->open
                    && close (ConnectionNumber (_xplotters[i]->x_dpy)) < 0
                    && errno != EINTR)
                /* shouldn't happen */
                _plotter->error (R___(_plotter)
                                 "the connection to the X display could not be closed");

        /* Repaint by sending an expose event to ourselves, copying the Label
        widget's background pixmap into its window.  This is a good idea
         because the window could have been resized during the
         openpl..closepl.  We don't do this if not double buffering (and
         presumably animating), unless the window size has changed since
         openpl was invoked (repainting makes the window flash, possibly
         irritating users). */
        if (_plotter->x_double_buffering != X_DBL_BUF_NONE)
            need_redisplay = true;
        else
        {
            Arg wargs[2];		/* werewolves */
            Dimension our_window_height, our_window_width;

#ifdef USE_MOTIF
            XtSetArg (wargs[0], XmNwidth, &our_window_width);
            XtSetArg (wargs[1], XmNheight, &our_window_height);
#else
            XtSetArg (wargs[0], XtNwidth, &our_window_width);
            XtSetArg (wargs[1], XtNheight, &our_window_height);
#endif
            XtGetValues (_plotter->y_canvas, wargs, (Cardinal)2);
            if ((_plotter->data->imax + 1 != (int)our_window_width)
                    || (_plotter->data->jmin + 1 != (int)our_window_height))
                /* window changed size */
                need_redisplay = true;
        }

        /* turn off backing store (if used); when we send the expose event to
        ourselves we want to repaint from the background pixmap, NOT from
         the server's backing store */
        {
            XSetWindowAttributes attributes;
            unsigned long value_mask;

            attributes.backing_store = NotUseful;
            value_mask = CWBackingStore;
            XChangeWindowAttributes (_plotter->x_dpy, (Window)_plotter->x_drawable2,
                                     value_mask, &attributes);
        }

        if (need_redisplay)
            /* send expose event to ourselves */
            XClearArea (_plotter->x_dpy,
                        (Window)_plotter->x_drawable2,
                        0, 0,
                        (unsigned int)0, (unsigned int)0,
                        True);

        _plotter->data->open = false; /* flag Plotter as closed (is this useful,
				       or just pedantic?) */

        /* Manage the window.  We won't get any events associated with other
        windows i.e. with previous invocations of openpl..closepl on this
         Plotter, or with other Plotters, since there's a distinct
         application context for every openpl..closepl. */
        XtAppMainLoop (_plotter->y_app_con); /* shouldn't return */

        /* NOTREACHED */
        exit (EXIT_FAILURE);
    }
}