#include <u.h> #include "x11-inc.h" #include <libc.h> #include <draw.h> #include <memdraw.h> #include "x11-memdraw.h" /* * Allocate a Memimage with an optional pixmap backing on the X server. */ Memimage* _xallocmemimage(Rectangle r, u32int chan, int pixmap) { int d, offset; Memimage *m; Xmem *xm; XImage *xi; m = _allocmemimage(r, chan); if(chan != GREY1 && chan != _x.chan) return m; /* * For bootstrapping, don't bother storing 1x1 images * on the X server. Memimageinit needs to allocate these * and we memimageinit before we do the rest of the X stuff. * Of course, 1x1 images on the server are useless anyway. */ if(Dx(r)==1 && Dy(r)==1) return m; xm = mallocz(sizeof(Xmem), 1); if(xm == nil){ freememimage(m); return nil; } /* * Allocate backing store. */ if(chan == GREY1) d = 1; else d = _x.depth; if(pixmap != PMundef) xm->pixmap = pixmap; else xm->pixmap = XCreatePixmap(_x.display, _x.drawable, Dx(r), Dy(r), d); /* * We want to align pixels on word boundaries. */ if(m->depth == 24) offset = r.min.x&3; else offset = r.min.x&(31/m->depth); r.min.x -= offset; assert(wordsperline(r, m->depth) <= m->width); /* * Wrap our data in an XImage structure. */ xi = XCreateImage(_x.display, _x.vis, d, ZPixmap, 0, (char*)m->data->bdata, Dx(r), Dy(r), 32, m->width*sizeof(u32int)); if(xi == nil){ freememimage(m); if(xm->pixmap != pixmap) XFreePixmap(_x.display, xm->pixmap); return nil; } xm->xi = xi; xm->r = r; /* * Set the XImage parameters so that it looks exactly like * a Memimage -- we're using the same data. */ if(m->depth < 8 || m->depth == 24) xi->bitmap_unit = 8; else xi->bitmap_unit = m->depth; xi->byte_order = LSBFirst; xi->bitmap_bit_order = MSBFirst; xi->bitmap_pad = 32; XInitImage(xi); XFlush(_x.display); m->X = xm; return m; } Memimage* allocmemimage(Rectangle r, u32int chan) { return _xallocmemimage(r, chan, PMundef); } void freememimage(Memimage *m) { Xmem *xm; if(m == nil) return; xm = m->X; if(xm && m->data->ref == 1){ if(xm->xi){ xm->xi->data = nil; XFree(xm->xi); } XFreePixmap(_x.display, xm->pixmap); free(xm); m->X = nil; } _freememimage(m); }