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
;
50 event_data
*event_fifo
;
51 int read_pos
, read_count
;
56 static void syborg_pointer_update(SyborgPointerState
*s
)
58 qemu_set_irq(s
->irq
, s
->read_count
&& s
->int_enabled
);
61 static uint64_t syborg_pointer_read(void *opaque
, target_phys_addr_t offset
,
64 SyborgPointerState
*s
= (SyborgPointerState
*)opaque
;
67 switch (offset
>> 2) {
69 return s
->absolute
? SYBORG_ID_TOUCHSCREEN
: SYBORG_ID_MOUSE
;
70 case POINTER_FIFO_COUNT
:
73 return s
->event_fifo
[s
->read_pos
].x
;
75 return s
->event_fifo
[s
->read_pos
].y
;
77 return s
->event_fifo
[s
->read_pos
].z
;
79 return s
->event_fifo
[s
->read_pos
].pointer_buttons
;
80 case POINTER_INT_ENABLE
:
81 return s
->int_enabled
;
82 case POINTER_FIFO_SIZE
:
85 cpu_abort(cpu_single_env
, "syborg_pointer_read: Bad offset %x\n",
91 static void syborg_pointer_write(void *opaque
, target_phys_addr_t offset
,
92 uint64_t value
, unsigned size
)
94 SyborgPointerState
*s
= (SyborgPointerState
*)opaque
;
97 switch (offset
>> 2) {
99 if (s
->read_count
> 0) {
101 if (++s
->read_pos
== s
->fifo_size
)
105 case POINTER_INT_ENABLE
:
106 s
->int_enabled
= value
;
109 cpu_abort(cpu_single_env
, "syborg_pointer_write: Bad offset %x\n",
112 syborg_pointer_update(s
);
115 static const MemoryRegionOps syborg_pointer_ops
= {
116 .read
= syborg_pointer_read
,
117 .write
= syborg_pointer_write
,
118 .endianness
= DEVICE_NATIVE_ENDIAN
,
121 static void syborg_pointer_event(void *opaque
, int dx
, int dy
, int dz
,
124 SyborgPointerState
*s
= (SyborgPointerState
*)opaque
;
125 int slot
= s
->read_pos
+ s
->read_count
;
127 /* This first FIFO entry is used to store current register state. */
128 if (s
->read_count
< s
->fifo_size
- 1) {
133 if (slot
>= s
->fifo_size
)
134 slot
-= s
->fifo_size
;
136 if (s
->read_count
== s
->fifo_size
&& !s
->absolute
) {
137 /* Merge existing entries. */
138 s
->event_fifo
[slot
].x
+= dx
;
139 s
->event_fifo
[slot
].y
+= dy
;
140 s
->event_fifo
[slot
].z
+= dz
;
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
].pointer_buttons
= buttons_state
;
148 syborg_pointer_update(s
);
151 static const VMStateDescription vmstate_event_data
= {
152 .name
= "dbma_channel",
154 .minimum_version_id
= 0,
155 .minimum_version_id_old
= 0,
156 .fields
= (VMStateField
[]) {
157 VMSTATE_INT32(x
, event_data
),
158 VMSTATE_INT32(y
, event_data
),
159 VMSTATE_INT32(z
, event_data
),
160 VMSTATE_INT32(pointer_buttons
, event_data
),
161 VMSTATE_END_OF_LIST()
165 static const VMStateDescription vmstate_syborg_pointer
= {
166 .name
= "syborg_pointer",
168 .minimum_version_id
= 1,
169 .minimum_version_id_old
= 1,
170 .fields
= (VMStateField
[]) {
171 VMSTATE_UINT32_EQUAL(fifo_size
, SyborgPointerState
),
172 VMSTATE_UINT32_EQUAL(absolute
, SyborgPointerState
),
173 VMSTATE_INT32(int_enabled
, SyborgPointerState
),
174 VMSTATE_INT32(read_pos
, SyborgPointerState
),
175 VMSTATE_INT32(read_count
, SyborgPointerState
),
176 VMSTATE_STRUCT_VARRAY_UINT32(event_fifo
, SyborgPointerState
, fifo_size
,
177 1, vmstate_event_data
, event_data
),
178 VMSTATE_END_OF_LIST()
182 static int syborg_pointer_init(SysBusDevice
*dev
)
184 SyborgPointerState
*s
= FROM_SYSBUS(SyborgPointerState
, dev
);
186 sysbus_init_irq(dev
, &s
->irq
);
187 memory_region_init_io(&s
->iomem
, &syborg_pointer_ops
, s
,
189 sysbus_init_mmio(dev
, &s
->iomem
);
191 if (s
->fifo_size
<= 0) {
192 fprintf(stderr
, "syborg_pointer: fifo too small\n");
195 s
->event_fifo
= g_malloc0(s
->fifo_size
* sizeof(s
->event_fifo
[0]));
197 qemu_add_mouse_event_handler(syborg_pointer_event
, s
, s
->absolute
,
200 vmstate_register(&dev
->qdev
, -1, &vmstate_syborg_pointer
, s
);
204 static SysBusDeviceInfo syborg_pointer_info
= {
205 .init
= syborg_pointer_init
,
206 .qdev
.name
= "syborg,pointer",
207 .qdev
.size
= sizeof(SyborgPointerState
),
208 .qdev
.props
= (Property
[]) {
209 DEFINE_PROP_UINT32("fifo-size", SyborgPointerState
, fifo_size
, 16),
210 DEFINE_PROP_UINT32("absolute", SyborgPointerState
, absolute
, 1),
211 DEFINE_PROP_END_OF_LIST(),
215 static void syborg_pointer_register_devices(void)
217 sysbus_register_withprop(&syborg_pointer_info
);
220 device_init(syborg_pointer_register_devices
)