2 * Syborg pointing device (mouse/touchscreen)
4 * Copyright (c) 2008 CodeSourcery
6 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 * of this software and associated documentation files (the "Software"), to deal
8 * in the Software without restriction, including without limitation the rights
9 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 * copies of the Software, and to permit persons to whom the Software is
11 * furnished to do so, subject to the following conditions:
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
32 POINTER_FIFO_COUNT
= 2,
37 POINTER_INT_ENABLE
= 7,
42 int x
, y
, z
, pointer_buttons
;
49 event_data
*event_fifo
;
50 int read_pos
, read_count
;
55 static void syborg_pointer_update(SyborgPointerState
*s
)
57 qemu_set_irq(s
->irq
, s
->read_count
&& s
->int_enabled
);
60 static uint32_t syborg_pointer_read(void *opaque
, target_phys_addr_t offset
)
62 SyborgPointerState
*s
= (SyborgPointerState
*)opaque
;
65 switch (offset
>> 2) {
67 return s
->absolute
? SYBORG_ID_TOUCHSCREEN
: SYBORG_ID_MOUSE
;
68 case POINTER_FIFO_COUNT
:
71 return s
->event_fifo
[s
->read_pos
].x
;
73 return s
->event_fifo
[s
->read_pos
].y
;
75 return s
->event_fifo
[s
->read_pos
].z
;
77 return s
->event_fifo
[s
->read_pos
].pointer_buttons
;
78 case POINTER_INT_ENABLE
:
79 return s
->int_enabled
;
80 case POINTER_FIFO_SIZE
:
83 cpu_abort(cpu_single_env
, "syborg_pointer_read: Bad offset %x\n",
89 static void syborg_pointer_write(void *opaque
, target_phys_addr_t offset
,
92 SyborgPointerState
*s
= (SyborgPointerState
*)opaque
;
95 switch (offset
>> 2) {
97 if (s
->read_count
> 0) {
99 if (++s
->read_pos
== s
->fifo_size
)
103 case POINTER_INT_ENABLE
:
104 s
->int_enabled
= value
;
107 cpu_abort(cpu_single_env
, "syborg_pointer_write: Bad offset %x\n",
110 syborg_pointer_update(s
);
113 static CPUReadMemoryFunc
* const syborg_pointer_readfn
[] = {
119 static CPUWriteMemoryFunc
* const syborg_pointer_writefn
[] = {
120 syborg_pointer_write
,
121 syborg_pointer_write
,
125 static void syborg_pointer_event(void *opaque
, int dx
, int dy
, int dz
,
128 SyborgPointerState
*s
= (SyborgPointerState
*)opaque
;
129 int slot
= s
->read_pos
+ s
->read_count
;
131 /* This first FIFO entry is used to store current register state. */
132 if (s
->read_count
< s
->fifo_size
- 1) {
137 if (slot
>= s
->fifo_size
)
138 slot
-= s
->fifo_size
;
140 if (s
->read_count
== s
->fifo_size
&& !s
->absolute
) {
141 /* Merge existing entries. */
142 s
->event_fifo
[slot
].x
+= dx
;
143 s
->event_fifo
[slot
].y
+= dy
;
144 s
->event_fifo
[slot
].z
+= dz
;
146 s
->event_fifo
[slot
].x
= dx
;
147 s
->event_fifo
[slot
].y
= dy
;
148 s
->event_fifo
[slot
].z
= dz
;
150 s
->event_fifo
[slot
].pointer_buttons
= buttons_state
;
152 syborg_pointer_update(s
);
155 static void syborg_pointer_save(QEMUFile
*f
, void *opaque
)
157 SyborgPointerState
*s
= (SyborgPointerState
*)opaque
;
160 qemu_put_be32(f
, s
->fifo_size
);
161 qemu_put_be32(f
, s
->absolute
);
162 qemu_put_be32(f
, s
->int_enabled
);
163 qemu_put_be32(f
, s
->read_pos
);
164 qemu_put_be32(f
, s
->read_count
);
165 for (i
= 0; i
< s
->fifo_size
; i
++) {
166 qemu_put_be32(f
, s
->event_fifo
[i
].x
);
167 qemu_put_be32(f
, s
->event_fifo
[i
].y
);
168 qemu_put_be32(f
, s
->event_fifo
[i
].z
);
169 qemu_put_be32(f
, s
->event_fifo
[i
].pointer_buttons
);
173 static int syborg_pointer_load(QEMUFile
*f
, void *opaque
, int version_id
)
175 SyborgPointerState
*s
= (SyborgPointerState
*)opaque
;
182 val
= qemu_get_be32(f
);
183 if (val
!= s
->fifo_size
)
186 val
= qemu_get_be32(f
);
187 if (val
!= s
->absolute
)
190 s
->int_enabled
= qemu_get_be32(f
);
191 s
->read_pos
= qemu_get_be32(f
);
192 s
->read_count
= qemu_get_be32(f
);
193 for (i
= 0; i
< s
->fifo_size
; i
++) {
194 s
->event_fifo
[i
].x
= qemu_get_be32(f
);
195 s
->event_fifo
[i
].y
= qemu_get_be32(f
);
196 s
->event_fifo
[i
].z
= qemu_get_be32(f
);
197 s
->event_fifo
[i
].pointer_buttons
= qemu_get_be32(f
);
202 static int syborg_pointer_init(SysBusDevice
*dev
)
204 SyborgPointerState
*s
= FROM_SYSBUS(SyborgPointerState
, dev
);
207 sysbus_init_irq(dev
, &s
->irq
);
208 iomemtype
= cpu_register_io_memory(syborg_pointer_readfn
,
209 syborg_pointer_writefn
, s
,
210 DEVICE_NATIVE_ENDIAN
);
211 sysbus_init_mmio(dev
, 0x1000, iomemtype
);
213 if (s
->fifo_size
<= 0) {
214 fprintf(stderr
, "syborg_pointer: fifo too small\n");
217 s
->event_fifo
= qemu_mallocz(s
->fifo_size
* sizeof(s
->event_fifo
[0]));
219 qemu_add_mouse_event_handler(syborg_pointer_event
, s
, s
->absolute
,
222 register_savevm(&dev
->qdev
, "syborg_pointer", -1, 1,
223 syborg_pointer_save
, syborg_pointer_load
, s
);
227 static SysBusDeviceInfo syborg_pointer_info
= {
228 .init
= syborg_pointer_init
,
229 .qdev
.name
= "syborg,pointer",
230 .qdev
.size
= sizeof(SyborgPointerState
),
231 .qdev
.props
= (Property
[]) {
232 DEFINE_PROP_UINT32("fifo-size", SyborgPointerState
, fifo_size
, 16),
233 DEFINE_PROP_UINT32("absolute", SyborgPointerState
, absolute
, 1),
234 DEFINE_PROP_END_OF_LIST(),
238 static void syborg_pointer_register_devices(void)
240 sysbus_register_withprop(&syborg_pointer_info
);
243 device_init(syborg_pointer_register_devices
)