⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 xwindow.c

📁 嵌入式linux下液晶编程院代码, 可以在LINUX2.4内核下运行
💻 C
字号:
/* $Id: XWindow.c,v 1.34 2003/09/10 14:01:53 reinelt Exp $ * * X11 Driver for LCD4Linux  * * (c) 2000 Herbert Rosmanith <herp@widsau.idv.uni-linz.ac.at> * * This program 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. * * This program 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.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	<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_UNDEFINEDunion semun {  int val;  struct semid_ds *buf;  unsigned short int *array;  struct seminfo *__buf;};#endifstatic 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 0200static 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("size","20x4"),"%dx%d",&cols,&rows)!=2      || rows<1 || cols<1) {    error ("X11: bad size '%s'",s);    return -1;  }  if (sscanf(s=cfg_get("font","5x8"),"%dx%d",&xres,&yres)!=2      || xres<5 || yres>10) {    error ("X11: bad font '%s'",s);    return -1;  }  if (sscanf(s=cfg_get("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("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("border", 0, 0, 1000000, &border)<0) return -1;  rgbfg=cfg_get("foreground","#000000");  rgbbg=cfg_get("background","#80d000");  rgbhg=cfg_get("halfground","#70c000");  if (*rgbfg=='\\') rgbfg++;  if (*rgbbg=='\\') rgbbg++;  if (*rgbhg=='\\') rgbhg++;  if (pix_init(rows,cols,xres,yres)==-1) return -1;  if (cfg_number("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) {  return pix_clear();}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 }};

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -