aboutsummaryrefslogtreecommitdiffstats
path: root/XWindow.c
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--XWindow.c628
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 }
-};