/* $Id: XWindow.c,v 1.39 2004/01/29 04:40:02 reinelt Exp $ * * X11 Driver for LCD4Linux * * Copyright 2000 Herbert Rosmanith <herp@wildsau.idv.uni-linz.ac.at> * * This file is part of LCD4Linux. * * LCD4Linux is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * LCD4Linux is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * * * $Log: XWindow.c,v $ * Revision 1.39 2004/01/29 04:40:02 reinelt * every .c file includes "config.h" now * * Revision 1.38 2004/01/09 04:16:06 reinelt * added 'section' argument to cfg_get(), but NULLed it on all calls by now. * * Revision 1.37 2004/01/06 22:33:14 reinelt * Copyright statements cleaned up * * Revision 1.36 2003/10/22 04:32:25 reinelt * fixed icon bug found by Rob van Nieuwkerk * * Revision 1.35 2003/10/05 17:58:50 reinelt * libtool junk; copyright messages cleaned up * * Revision 1.34 2003/09/10 14:01:53 reinelt * icons nearly finished\! * * Revision 1.33 2003/09/09 06:54:43 reinelt * new function 'cfg_number()' * * Revision 1.32 2003/07/24 04:48:09 reinelt * 'soft clear' needed for virtual rows * * Revision 1.31 2003/04/12 16:23:10 reinelt * small glitch in XWindow.c (thanks to Moe Wibble) * * Revision 1.30 2003/02/22 07:53:10 reinelt * cfg_get(key,defval) * * Revision 1.29 2003/02/18 06:13:44 reinelt * X11 driver fixes and cleanup * * Revision 1.28 2003/02/17 06:06:12 reinelt * small bug in X11 driver: omit pixel gap between cahracters * * Revision 1.27 2002/08/19 04:41:20 reinelt * introduced bar.c, moved bar stuff from display.h to bar.h * * Revision 1.26 2001/08/05 17:13:29 reinelt * * cleaned up inlude of sys/time.h and time.h * * Revision 1.25 2001/03/16 16:40:17 ltoetsch * implemented time bar * * Revision 1.24 2001/03/01 11:08:16 reinelt * * reworked configure to allow selection of drivers * * Revision 1.23 2001/02/26 00:33:37 herp * fixed X11 signal handler * * Revision 1.22 2001/02/13 09:00:13 reinelt * * prepared framework for GPO's (general purpose outputs) * * Revision 1.21 2000/08/10 18:42:20 reinelt * * fixed some bugs with the new syslog code * * Revision 1.20 2000/08/10 09:44:09 reinelt * * new debugging scheme: error(), info(), debug() * uses syslog if in daemon mode * * Revision 1.19 2000/08/09 09:50:29 reinelt * * opened 0.98 development * removed driver-specific signal-handlers * added 'quit'-function to driver structure * added global signal-handler * * Revision 1.18 2000/05/02 23:07:48 herp * Crystalfontz initial coding * * Revision 1.17 2000/04/05 05:58:36 reinelt * * fixed bug in XWindow.c: union semun isn't defined with glibc-2.1 * * Revision 1.16 2000/04/03 23:53:23 herp * fixed a bug that caused pixel-errors ("fliegendreck") under high load * * Revision 1.15 2000/04/03 04:01:31 reinelt * * if 'gap' is specified as -1, a gap of (pixelsize+pixelgap) is selected automatically * * Revision 1.14 2000/04/02 22:07:10 herp * fixded a bug that occasionally caused Xlib errors * * Revision 1.13 2000/04/01 22:40:42 herp * geometric correction (too many pixelgaps) * lcd4linux main should return int, not void * * Revision 1.12 2000/04/01 19:33:45 herp * * colors in format \#RRGGBB in config-file now understood * * Revision 1.11 2000/04/01 16:22:38 reinelt * * bug that caused a segfault in processor.c fixed (thanks to herp) * * Revision 1.10 2000/03/31 01:42:11 herp * * semaphore bug fixed * * Revision 1.9 2000/03/30 16:46:57 reinelt * * configure now handles '--with-x' and '--without-x' correct * * Revision 1.8 2000/03/28 08:48:33 reinelt * * README.X11 added * * Revision 1.7 2000/03/28 07:22:15 reinelt * * version 0.95 released * X11 driver up and running * minor bugs fixed * */ /* * * exported fuctions: * * struct LCD XWindow[] * */ /* * Mon Feb 26 02:07:52 MET 2001 fixed sighandler * Tue Apr 4 02:37:38 MET 2000 fixed a bug that caused pixelerrors under h/load * Sun Apr 2 22:07:10 MET 2000 fixed a bug that occasionally caused Xlib error * Sun Apr 2 01:32:48 MET 2000 geometric correction (too many pixelgaps) * Sat Apr 1 22:18:04 MET 2000 colors in format \#RRGGBB in config-file * Fri Mar 31 01:42:11 MET 2000 semaphore bug fixed * Sun Mar 26 15:28:23 MET 2000 various rewrites * Sat Mar 25 23:58:19 MET 2000 use generic pixmap driver * Thu Mar 23 01:05:07 MET 2000 multithreading, synchronization * Tue Mar 21 22:22:03 MET 2000 initial coding * */ #include "config.h" #include <X11/Xlib.h> #include <X11/Xutil.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <errno.h> #include <sys/types.h> #include <sys/wait.h> #include <sys/ipc.h> #include <sys/sem.h> #include <sys/shm.h> #include <unistd.h> #include <signal.h> #include "debug.h" #include "cfg.h" #include "display.h" #include "bar.h" #include "icon.h" #include "pixmap.h" /* glibc 2.1 requires defining semun ourselves */ #ifdef _SEM_SEMUN_UNDEFINED union semun { int val; struct semid_ds *buf; unsigned short int *array; struct seminfo *__buf; }; #endif static LCD Lcd; static Display *dp; static int sc; static Window w,rw; static Visual *vi; static int dd; static Colormap cm; static GC gc,gcb,gch; static XColor co[3]; static Pixmap pmback; static unsigned char *LCDpixmap2; static char *rgbfg,*rgbbg,*rgbhg; static int pixel=-1; /*pointsize in pixel*/ static int pgap=0; /*gap between points */ static int rgap=0; /*row gap between lines*/ static int cgap=0; /*column gap between characters*/ static int border=0; /*window border*/ static int rows=-1,cols=-1; /*rows+cols without background*/ static int xres=-1,yres=-1; /*xres+yres (same as self->...)*/ static int icons; /* number of user-defined icons */ static int dimx,dimy; /*total window dimension in pixel*/ static int boxw,boxh; /*box width, box height*/ static int async_update(); /*PROTO*/ static pid_t async_updater_pid=1; static int semid=-1; static int shmid=-1; static int ppid; /*parent pid*/ static void acquire_lock() { struct sembuf sembuf; sembuf.sem_num=0; sembuf.sem_op=-1; sembuf.sem_flg=0; semop(semid,&sembuf,1); /* get mutex */ } static void release_lock() { struct sembuf sembuf; sembuf.sem_num=0; sembuf.sem_op=1; sembuf.sem_flg=0; semop(semid,&sembuf,1); /* free mutex */ } static void semcleanup() { union semun arg; if (semid>-1) semctl(semid,0,IPC_RMID,arg); } static void shmcleanup() { if (shmid>-1) shmctl(shmid,IPC_RMID,NULL); } static void quit_updater() { int i; if (async_updater_pid>1) { i=async_updater_pid; async_updater_pid=1; kill(i,9); waitpid(i,&i,0); } } static void sig_updater(int sig) { kill(ppid,sig); exit(0); } static void init_signals() { unsigned int ignsig=(1<<SIGBUS)|(1<<SIGFPE)|(1<<SIGSEGV)| (1<<SIGTSTP)|(1<<SIGCHLD)|(1<<SIGCONT)| (1<<SIGTTIN)|(1<<SIGWINCH); int i; for(i=0;i<NSIG;i++) if (((1<<i)&ignsig)==0) signal(i,sig_updater); } static int init_shm(int nbytes,unsigned char **buf) { shmid=shmget(IPC_PRIVATE,nbytes,SHM_R|SHM_W); if (shmid==-1) { error ("X11: shmget() failed: %s", strerror(errno)); return -1; } *buf=shmat(shmid,NULL,0); if (*buf==NULL) { error ("X11: shmat() failed: %s", strerror(errno)); return -1; } return 0; } /* acording to SUN-Solaris man-pages: */ #define SEM_ALTER 0200 static int init_thread(int bufsiz) { union semun semun; semid=semget(IPC_PRIVATE,1,SEM_ALTER); if (semid==-1) { error ("X11: semget() failed: %s", strerror(errno)); return -1; } semun.val=1; semctl(semid,0,SETVAL,semun); ppid=getpid(); switch(async_updater_pid=fork()) { case -1: error ("X11: fork() failed: %s", strerror(errno)); return -1; case 0: async_update(); error ("X11: async_update failed"); kill(ppid,SIGTERM); exit(-1); default: break; } signal(SIGCHLD,quit_updater); atexit(quit_updater); return 0; } static int init_x(int rows,int cols,int xres,int yres) { XSetWindowAttributes wa; XSizeHints size_hints; XColor co_dummy; XEvent ev; if ((dp=XOpenDisplay(NULL))==NULL) { error ("X11: can't open display"); return -1; } sc=DefaultScreen(dp); gc=DefaultGC(dp,sc); vi=DefaultVisual(dp,sc); dd=DefaultDepth(dp,sc); rw=DefaultRootWindow(dp); cm=DefaultColormap(dp,sc); if (XAllocNamedColor(dp,cm,rgbfg,&co[0],&co_dummy)==False) { error ("X11: can't alloc foreground color '%s'", rgbfg); return -1; } if (XAllocNamedColor(dp,cm,rgbbg,&co[1],&co_dummy)==False) { error ("X11: can't alloc background color '%s'", rgbbg); return -1; } if (XAllocNamedColor(dp,cm,rgbhg,&co[2],&co_dummy)==False) { error ("X11: can't alloc halfground color '%s'", rgbhg); return -1; } boxw=xres*pixel+(xres-1)*pgap+cgap; boxh=yres*pixel+(yres-1)*pgap+rgap; dimx=cols*xres*pixel+(cols*xres-1)*pgap+(cols-1)*cgap; dimy=rows*yres*pixel+(rows*yres-1)*pgap+(rows-1)*rgap; wa.event_mask=ExposureMask; w=XCreateWindow(dp,rw,0,0,dimx+2*border,dimy+2*border,0,0, InputOutput,vi,CWEventMask,&wa); pmback=XCreatePixmap(dp,w,dimx,dimy,dd); size_hints.min_width=size_hints.max_width=dimx+2*border; size_hints.min_height=size_hints.max_height=dimy+2*border; size_hints.flags=PPosition|PSize|PMinSize|PMaxSize; XSetWMProperties(dp,w,NULL,NULL,NULL,0,&size_hints,NULL,NULL); XSetForeground(dp,gc,co[0].pixel); XSetBackground(dp,gc,co[1].pixel); gcb=XCreateGC(dp,w,0,NULL); XSetForeground(dp,gcb,co[1].pixel); XSetBackground(dp,gcb,co[0].pixel); gch=XCreateGC(dp,w,0,NULL); XSetForeground(dp,gch,co[2].pixel); XSetBackground(dp,gch,co[0].pixel); XFillRectangle(dp,pmback,gcb,0,0,dimx,dimy); XSetWindowBackground(dp,w,co[1].pixel); XClearWindow(dp,w); XStoreName(dp,w,"XLCD4Linux"); XMapWindow(dp,w); XFlush(dp); for(;;) { XNextEvent(dp,&ev); if (ev.type==Expose && ev.xexpose.count==0) break; } return 0; } int xlcdinit(LCD *Self) { char *s; if (sscanf(s=cfg_get(NULL, "size", "20x4"),"%dx%d",&cols,&rows)!=2 || rows<1 || cols<1) { error ("X11: bad size '%s'",s); return -1; } if (sscanf(s=cfg_get(NULL, "font", "5x8"),"%dx%d",&xres,&yres)!=2 || xres<5 || yres>10) { error ("X11: bad font '%s'",s); return -1; } if (sscanf(s=cfg_get(NULL, "pixel", "4+1"),"%d+%d",&pixel,&pgap)!=2 || pixel<1 || pgap<0) { error ("X11: bad pixel '%s'",s); return -1; } if (sscanf(s=cfg_get(NULL, "gap", "-1x-1"),"%dx%d",&cgap,&rgap)!=2 || cgap<-1 || rgap<-1) { error ("X11: bad gap '%s'",s); return -1; } if (rgap<0) rgap=pixel+pgap; if (cgap<0) cgap=pixel+pgap; if (cfg_number(NULL, "border", 0, 0, 1000000, &border)<0) return -1; rgbfg=cfg_get(NULL, "foreground", "#000000"); rgbbg=cfg_get(NULL, "background", "#80d000"); rgbhg=cfg_get(NULL, "halfground", "#70c000"); if (*rgbfg=='\\') rgbfg++; if (*rgbbg=='\\') rgbbg++; if (*rgbhg=='\\') rgbhg++; if (pix_init(rows,cols,xres,yres)==-1) return -1; if (cfg_number(NULL, "Icons", 0, 0, 8, &icons) < 0) return -1; if (icons>0) { info ("allocating %d icons", icons); icon_init(rows, cols, xres, yres, 8, icons, pix_icon); } if (init_x(rows,cols,xres,yres)==-1) return -1; init_signals(); if (init_shm(rows*cols*xres*yres,&LCDpixmap2)==-1) return -1; memset(LCDpixmap2,0xff,rows*yres*cols*xres); if (init_thread(rows*cols*xres*yres)==-1) return -1; Self->rows=rows; Self->cols=cols; Self->xres=xres; Self->yres=yres; Self->icons=icons; Lcd=*Self; pix_clear(); return 0; } int xlcdclear(int full) { icon_clear(); pix_clear(); return 0; } int xlcdput(int row,int col, char *text) { return pix_put(row,col,text); } int xlcdbar(int type, int row, int col, int max, int len1, int len2) { return pix_bar(type,row,col,max,len1,len2); } int xlcdicon (int num, int seq, int row, int col) { return icon_draw (num, seq, row, col); } int xlcdflush() { int dirty; int row,col; acquire_lock(); dirty=0; for(row=0;row<rows*yres;row++) { int y=border+(row/yres)*rgap+row*(pixel+pgap); for(col=0;col<cols*xres;col++) { int x=border+(col/xres)*cgap+col*(pixel+pgap); int p=row*cols*xres+col; if (LCDpixmap[p]^LCDpixmap2[p]) { XFillRectangle(dp,w,LCDpixmap[p]?gc:gch,x,y,pixel,pixel); LCDpixmap2[p]=LCDpixmap[p]; dirty=1; } } } if (dirty) XSync(dp,False); release_lock(); return 0; } int xlcdquit(void) { semcleanup(); shmcleanup(); quit_updater(); return 0; } /* * this one should only be called from the updater-thread * no user serviceable parts inside */ static void update(int x,int y,int width,int height) { /* * theory of operation: * instead of the old, fully-featured but complicated update * region calculation, we do an update of the whole display, * but check before every pixel if the pixel region is inside * the update region. */ int x0, y0; int x1, y1; int row, col; int dirty; x0=x-pixel; y0=y-pixel; x1=x+pixel+width; y1=y+pixel+height; dirty=0; for(row=0;row<rows*yres;row++) { int y=border+(row/yres)*rgap+row*(pixel+pgap); if (y<y0 || y>y1) continue; for(col=0;col<cols*xres;col++) { int x=border+(col/xres)*cgap+col*(pixel+pgap); int p; if (x<x0 || x>x1) continue; p=row*cols*xres+col; XFillRectangle(dp,w,LCDpixmap2[p]?gc:gch,x,y,pixel,pixel); dirty=1; } } if (dirty) XSync(dp,False); } static int async_update() { XSetWindowAttributes wa; XEvent ev; if ((dp=XOpenDisplay(NULL))==NULL) return -1; wa.event_mask=ExposureMask; XChangeWindowAttributes(dp,w,CWEventMask,&wa); for(;;) { XWindowEvent(dp,w,ExposureMask,&ev); if (ev.type==Expose) { acquire_lock(); update(ev.xexpose.x,ev.xexpose.y, ev.xexpose.width,ev.xexpose.height); release_lock(); } } } LCD XWindow[] = { { name: "X11", rows: 0, cols: 0, xres: 0, yres: 0, bars: BAR_L | BAR_R | BAR_U | BAR_D | BAR_H2 | BAR_V2 | BAR_T, icons: 0, gpos: 0, init: xlcdinit, clear: xlcdclear, put: xlcdput, bar: xlcdbar, icon: xlcdicon, gpo: NULL, flush: xlcdflush, quit: xlcdquit }, { NULL } };