diff options
author | reinelt <reinelt@3ae390bd-cb1e-0410-b409-cd5a39f66f1f> | 2004-02-24 05:55:04 +0000 |
---|---|---|
committer | reinelt <reinelt@3ae390bd-cb1e-0410-b409-cd5a39f66f1f> | 2004-02-24 05:55:04 +0000 |
commit | 410645dbeaaae83a2f94b4396ac69759e205092c (patch) | |
tree | 753366deb77273164e34314735ef4d3ce5499701 /XWindow.c | |
parent | 6c9b3154be6c2cf216f4a35be384e7c4a36cc4bd (diff) | |
download | lcd4linux-410645dbeaaae83a2f94b4396ac69759e205092c.tar.gz |
[lcd4linux @ 2004-02-24 05:54:57 by reinelt]
X11 driver ported
git-svn-id: https://ssl.bulix.org/svn/lcd4linux/trunk@375 3ae390bd-cb1e-0410-b409-cd5a39f66f1f
Diffstat (limited to 'XWindow.c')
-rw-r--r-- | XWindow.c | 628 |
1 files changed, 0 insertions, 628 deletions
diff --git a/XWindow.c b/XWindow.c deleted file mode 100644 index c7ead72..0000000 --- a/XWindow.c +++ /dev/null @@ -1,628 +0,0 @@ -/* $Id: XWindow.c,v 1.40 2004/01/30 20:57:55 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.40 2004/01/30 20:57:55 reinelt - * HD44780 patch from Martin Hejl - * dmalloc integrated - * - * 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" - -#ifdef WITH_DMALLOC -#include <dmalloc.h> -#endif - - -/* 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 } -}; |