xref: /freebsd/lib/libvgl/mouse.c (revision 1b6c76a2fe091c74f08427e6c870851025a9cf67)
1 /*-
2  * Copyright (c) 1991-1997 S�ren Schmidt
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer
10  *    in this position and unchanged.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  *    notice, this list of conditions and the following disclaimer in the
13  *    documentation and/or other materials provided with the distribution.
14  * 3. The name of the author may not be used to endorse or promote products
15  *    derived from this software withough specific prior written permission
16  *
17  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  *
28  * $FreeBSD$
29  */
30 
31 #include <stdio.h>
32 #include <sys/types.h>
33 #include <sys/ioctl.h>
34 #include <sys/signal.h>
35 #include <sys/consio.h>
36 #include <sys/fbio.h>
37 #include "vgl.h"
38 
39 #define X 0xff
40 static byte StdAndMask[MOUSE_IMG_SIZE*MOUSE_IMG_SIZE] = {
41 	X,X,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
42 	X,X,X,0,0,0,0,0,0,0,0,0,0,0,0,0,
43 	X,X,X,X,0,0,0,0,0,0,0,0,0,0,0,0,
44 	X,X,X,X,X,0,0,0,0,0,0,0,0,0,0,0,
45 	X,X,X,X,X,X,0,0,0,0,0,0,0,0,0,0,
46 	X,X,X,X,X,X,X,0,0,0,0,0,0,0,0,0,
47 	X,X,X,X,X,X,X,X,0,0,0,0,0,0,0,0,
48 	X,X,X,X,X,X,X,X,X,0,0,0,0,0,0,0,
49 	X,X,X,X,X,X,X,0,0,0,0,0,0,0,0,0,
50 	0,0,0,X,X,X,X,0,0,0,0,0,0,0,0,0,
51 	0,0,0,X,X,X,X,X,0,0,0,0,0,0,0,0,
52 	0,0,0,0,X,X,X,X,0,0,0,0,0,0,0,0,
53 	0,0,0,0,X,X,X,X,0,0,0,0,0,0,0,0,
54 	0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
55 	0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
56 	0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
57 };
58 static byte StdOrMask[MOUSE_IMG_SIZE*MOUSE_IMG_SIZE] = {
59 	0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
60 	0,X,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
61 	0,X,X,0,0,0,0,0,0,0,0,0,0,0,0,0,
62 	0,X,X,X,0,0,0,0,0,0,0,0,0,0,0,0,
63 	0,X,X,X,X,0,0,0,0,0,0,0,0,0,0,0,
64 	0,X,X,X,X,X,0,0,0,0,0,0,0,0,0,0,
65 	0,X,X,X,X,X,X,0,0,0,0,0,0,0,0,0,
66 	0,X,X,0,X,0,0,0,0,0,0,0,0,0,0,0,
67 	0,0,0,0,X,X,0,0,0,0,0,0,0,0,0,0,
68 	0,0,0,0,X,X,0,0,0,0,0,0,0,0,0,0,
69 	0,0,0,0,0,X,X,0,0,0,0,0,0,0,0,0,
70 	0,0,0,0,0,X,X,0,0,0,0,0,0,0,0,0,
71 	0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
72 	0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
73 	0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
74 	0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
75 };
76 #undef X
77 static VGLBitmap VGLMouseStdAndMask =
78     VGLBITMAP_INITIALIZER(MEMBUF, MOUSE_IMG_SIZE, MOUSE_IMG_SIZE, StdAndMask);
79 static VGLBitmap VGLMouseStdOrMask =
80     VGLBITMAP_INITIALIZER(MEMBUF, MOUSE_IMG_SIZE, MOUSE_IMG_SIZE, StdOrMask);
81 static VGLBitmap *VGLMouseAndMask, *VGLMouseOrMask;
82 static byte map[MOUSE_IMG_SIZE*MOUSE_IMG_SIZE];
83 static VGLBitmap VGLMouseSave =
84     VGLBITMAP_INITIALIZER(MEMBUF, MOUSE_IMG_SIZE, MOUSE_IMG_SIZE, map);
85 static int VGLMouseVisible = 0;
86 static int VGLMouseFrozen = 0;
87 static int VGLMouseShown = 0;
88 static int VGLMouseXpos = 0;
89 static int VGLMouseYpos = 0;
90 static int VGLMouseButtons = 0;
91 
92 void
93 VGLMousePointerShow()
94 {
95   byte buf[MOUSE_IMG_SIZE*MOUSE_IMG_SIZE];
96   VGLBitmap buffer =
97     VGLBITMAP_INITIALIZER(MEMBUF, MOUSE_IMG_SIZE, MOUSE_IMG_SIZE, buf);
98   byte crtcidx, crtcval, gdcidx, gdcval;
99   int pos;
100 
101   if (!VGLMouseVisible) {
102     VGLMouseVisible = 1;
103     crtcidx = inb(0x3c4);
104     crtcval = inb(0x3c5);
105     gdcidx = inb(0x3ce);
106     gdcval = inb(0x3cf);
107     __VGLBitmapCopy(VGLDisplay, VGLMouseXpos, VGLMouseYpos,
108 		  &VGLMouseSave, 0, 0, MOUSE_IMG_SIZE, MOUSE_IMG_SIZE);
109     bcopy(VGLMouseSave.Bitmap, buffer.Bitmap, MOUSE_IMG_SIZE*MOUSE_IMG_SIZE);
110     for (pos = 0; pos <  MOUSE_IMG_SIZE*MOUSE_IMG_SIZE; pos++)
111       buffer.Bitmap[pos]=(buffer.Bitmap[pos]&~(VGLMouseAndMask->Bitmap[pos])) |
112 			   VGLMouseOrMask->Bitmap[pos];
113     __VGLBitmapCopy(&buffer, 0, 0, VGLDisplay,
114 		  VGLMouseXpos, VGLMouseYpos, MOUSE_IMG_SIZE, MOUSE_IMG_SIZE);
115     outb(0x3c4, crtcidx);
116     outb(0x3c5, crtcval);
117     outb(0x3ce, gdcidx);
118     outb(0x3cf, gdcval);
119   }
120 }
121 
122 void
123 VGLMousePointerHide()
124 {
125   byte crtcidx, crtcval, gdcidx, gdcval;
126 
127   if (VGLMouseVisible) {
128     VGLMouseVisible = 0;
129     crtcidx = inb(0x3c4);
130     crtcval = inb(0x3c5);
131     gdcidx = inb(0x3ce);
132     gdcval = inb(0x3cf);
133     __VGLBitmapCopy(&VGLMouseSave, 0, 0, VGLDisplay,
134 		  VGLMouseXpos, VGLMouseYpos, MOUSE_IMG_SIZE, MOUSE_IMG_SIZE);
135     outb(0x3c4, crtcidx);
136     outb(0x3c5, crtcval);
137     outb(0x3ce, gdcidx);
138     outb(0x3cf, gdcval);
139   }
140 }
141 
142 void
143 VGLMouseMode(int mode)
144 {
145   if (mode == VGL_MOUSESHOW) {
146     if (VGLMouseShown == VGL_MOUSEHIDE) {
147       VGLMousePointerShow();
148       VGLMouseShown = VGL_MOUSESHOW;
149     }
150   }
151   else {
152     if (VGLMouseShown == VGL_MOUSESHOW) {
153       VGLMousePointerHide();
154       VGLMouseShown = VGL_MOUSEHIDE;
155     }
156   }
157 }
158 
159 void
160 VGLMouseAction(int dummy)
161 {
162   struct mouse_info mouseinfo;
163 
164   if (VGLMouseFrozen) {
165     VGLMouseFrozen++;
166     return;
167   }
168   mouseinfo.operation = MOUSE_GETINFO;
169   ioctl(0, CONS_MOUSECTL, &mouseinfo);
170   if (VGLMouseShown == VGL_MOUSESHOW)
171     VGLMousePointerHide();
172   VGLMouseXpos = mouseinfo.u.data.x;
173   VGLMouseYpos = mouseinfo.u.data.y;
174   VGLMouseButtons = mouseinfo.u.data.buttons;
175   if (VGLMouseShown == VGL_MOUSESHOW)
176     VGLMousePointerShow();
177 }
178 
179 void
180 VGLMouseSetImage(VGLBitmap *AndMask, VGLBitmap *OrMask)
181 {
182   if (VGLMouseShown == VGL_MOUSESHOW)
183     VGLMousePointerHide();
184   VGLMouseAndMask = AndMask;
185   VGLMouseOrMask = OrMask;
186   if (VGLMouseShown == VGL_MOUSESHOW)
187     VGLMousePointerShow();
188 }
189 
190 void
191 VGLMouseSetStdImage()
192 {
193   if (VGLMouseShown == VGL_MOUSESHOW)
194     VGLMousePointerHide();
195   VGLMouseAndMask = &VGLMouseStdAndMask;
196   VGLMouseOrMask = &VGLMouseStdOrMask;
197   if (VGLMouseShown == VGL_MOUSESHOW)
198     VGLMousePointerShow();
199 }
200 
201 int
202 VGLMouseInit(int mode)
203 {
204   struct mouse_info mouseinfo;
205   int error;
206 
207   VGLMouseSetStdImage();
208   mouseinfo.operation = MOUSE_MODE;
209   mouseinfo.u.mode.signal = SIGUSR2;
210   if ((error = ioctl(0, CONS_MOUSECTL, &mouseinfo)))
211     return error;
212   signal(SIGUSR2, VGLMouseAction);
213   mouseinfo.operation = MOUSE_GETINFO;
214   ioctl(0, CONS_MOUSECTL, &mouseinfo);
215   VGLMouseXpos = mouseinfo.u.data.x;
216   VGLMouseYpos = mouseinfo.u.data.y;
217   VGLMouseButtons = mouseinfo.u.data.buttons;
218   VGLMouseMode(mode);
219   return 0;
220 }
221 
222 int
223 VGLMouseStatus(int *x, int *y, char *buttons)
224 {
225   signal(SIGUSR2, SIG_IGN);
226   *x =  VGLMouseXpos;
227   *y =  VGLMouseYpos;
228   *buttons =  VGLMouseButtons;
229   signal(SIGUSR2, VGLMouseAction);
230   return VGLMouseShown;
231 }
232 
233 int
234 VGLMouseFreeze(int x, int y, int width, int hight, byte color)
235 {
236   if (!VGLMouseFrozen) {
237     VGLMouseFrozen = 1;
238     if (width > 1 || hight > 1) {		/* bitmap */
239       if (VGLMouseShown == 1) {
240         int overlap;
241 
242         if (x > VGLMouseXpos)
243           overlap = (VGLMouseXpos + MOUSE_IMG_SIZE) - x;
244         else
245           overlap = (x + width) - VGLMouseXpos;
246         if (overlap > 0) {
247           if (y > VGLMouseYpos)
248             overlap = (VGLMouseYpos + MOUSE_IMG_SIZE) - y;
249           else
250             overlap = (y + hight) - VGLMouseYpos;
251           if (overlap > 0)
252             VGLMousePointerHide();
253         }
254       }
255     }
256     else {				/* bit */
257       if (VGLMouseShown &&
258           x >= VGLMouseXpos && x < VGLMouseXpos + MOUSE_IMG_SIZE &&
259           y >= VGLMouseYpos && y < VGLMouseYpos + MOUSE_IMG_SIZE) {
260         VGLMouseSave.Bitmap[(y-VGLMouseYpos)*MOUSE_IMG_SIZE+(x-VGLMouseXpos)] =
261           (color);
262         if (VGLMouseAndMask->Bitmap
263           [(y-VGLMouseYpos)*MOUSE_IMG_SIZE+(x-VGLMouseXpos)]) {
264           return 1;
265         }
266       }
267     }
268   }
269   return 0;
270 }
271 
272 void
273 VGLMouseUnFreeze()
274 {
275   if (VGLMouseFrozen > 1) {
276     VGLMouseFrozen = 0;
277     VGLMouseAction(0);
278   }
279   else {
280     VGLMouseFrozen = 0;
281     if (VGLMouseShown == VGL_MOUSESHOW && !VGLMouseVisible)
282       VGLMousePointerShow();
283   }
284 }
285