1 /* Copyright (C) 2007-2008 The Android Open Source Project
2 **
3 ** This software is licensed under the terms of the GNU General Public
4 ** License version 2, as published by the Free Software Foundation, and
5 ** may be copied, distributed, and modified under those terms.
6 **
7 ** This program is distributed in the hope that it will be useful,
8 ** but WITHOUT ANY WARRANTY; without even the implied warranty of
9 ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 ** GNU General Public License for more details.
11 */
12 #include "migration/qemu-file.h"
13 #include "sysemu/char.h"
14 #include "hw/android/goldfish/device.h"
15 #include "hw/android/goldfish/vmem.h"
16 #include "hw/hw.h"
17
18 enum {
19 TTY_PUT_CHAR = 0x00,
20 TTY_BYTES_READY = 0x04,
21 TTY_CMD = 0x08,
22
23 TTY_DATA_PTR = 0x10,
24 TTY_DATA_LEN = 0x14,
25 TTY_DATA_PTR_HIGH = 0x18,
26
27 TTY_CMD_INT_DISABLE = 0,
28 TTY_CMD_INT_ENABLE = 1,
29 TTY_CMD_WRITE_BUFFER = 2,
30 TTY_CMD_READ_BUFFER = 3,
31 };
32
33 struct tty_state {
34 struct goldfish_device dev;
35 CharDriverState *cs;
36 uint64_t ptr;
37 uint32_t ptr_len;
38 uint32_t ready;
39 uint8_t data[128];
40 uint32_t data_count;
41 };
42
43 #define GOLDFISH_TTY_SAVE_VERSION 2
44
goldfish_tty_save(QEMUFile * f,void * opaque)45 static void goldfish_tty_save(QEMUFile* f, void* opaque)
46 {
47 struct tty_state* s = opaque;
48
49 qemu_put_be64( f, s->ptr );
50 qemu_put_be32( f, s->ptr_len );
51 qemu_put_byte( f, s->ready );
52 qemu_put_byte( f, s->data_count );
53 qemu_put_buffer( f, s->data, s->data_count );
54 }
55
goldfish_tty_load(QEMUFile * f,void * opaque,int version_id)56 static int goldfish_tty_load(QEMUFile* f, void* opaque, int version_id)
57 {
58 struct tty_state* s = opaque;
59
60 if ((version_id != GOLDFISH_TTY_SAVE_VERSION) &&
61 (version_id != (GOLDFISH_TTY_SAVE_VERSION - 1))) {
62 return -1;
63 }
64 if (version_id == (GOLDFISH_TTY_SAVE_VERSION - 1)) {
65 s->ptr = (uint64_t)qemu_get_be32(f);
66 } else {
67 s->ptr = qemu_get_be64(f);
68 }
69 s->ptr_len = qemu_get_be32(f);
70 s->ready = qemu_get_byte(f);
71 s->data_count = qemu_get_byte(f);
72 qemu_get_buffer(f, s->data, s->data_count);
73
74 return 0;
75 }
76
goldfish_tty_read(void * opaque,hwaddr offset)77 static uint32_t goldfish_tty_read(void *opaque, hwaddr offset)
78 {
79 struct tty_state *s = (struct tty_state *)opaque;
80
81 //printf("goldfish_tty_read %x %x\n", offset, size);
82
83 switch (offset) {
84 case TTY_BYTES_READY:
85 return s->data_count;
86 default:
87 cpu_abort (cpu_single_env, "goldfish_tty_read: Bad offset %x\n", offset);
88 return 0;
89 }
90 }
91
goldfish_tty_write(void * opaque,hwaddr offset,uint32_t value)92 static void goldfish_tty_write(void *opaque, hwaddr offset, uint32_t value)
93 {
94 struct tty_state *s = (struct tty_state *)opaque;
95
96 //printf("goldfish_tty_write %x %x %x\n", offset, value, size);
97
98 switch(offset) {
99 case TTY_PUT_CHAR: {
100 uint8_t ch = value;
101 if(s->cs)
102 qemu_chr_write(s->cs, &ch, 1);
103 } break;
104
105 case TTY_CMD:
106 switch(value) {
107 case TTY_CMD_INT_DISABLE:
108 if(s->ready) {
109 if(s->data_count > 0)
110 goldfish_device_set_irq(&s->dev, 0, 0);
111 s->ready = 0;
112 }
113 break;
114
115 case TTY_CMD_INT_ENABLE:
116 if(!s->ready) {
117 if(s->data_count > 0)
118 goldfish_device_set_irq(&s->dev, 0, 1);
119 s->ready = 1;
120 }
121 break;
122
123 case TTY_CMD_WRITE_BUFFER:
124 if(s->cs) {
125 int len;
126 target_ulong buf;
127
128 buf = s->ptr;
129 len = s->ptr_len;
130
131 while (len) {
132 char temp[64];
133 int to_write = sizeof(temp);
134 if (to_write > len)
135 to_write = len;
136
137 safe_memory_rw_debug(current_cpu, buf, (uint8_t*)temp, to_write, 0);
138 qemu_chr_write(s->cs, (const uint8_t*)temp, to_write);
139 buf += to_write;
140 len -= to_write;
141 }
142 //printf("goldfish_tty_write: got %d bytes from %llx\n", s->ptr_len, (unsigned long long)s->ptr);
143 }
144 break;
145
146 case TTY_CMD_READ_BUFFER:
147 if(s->ptr_len > s->data_count)
148 cpu_abort (cpu_single_env, "goldfish_tty_write: reading more data than available %d %d\n", s->ptr_len, s->data_count);
149 safe_memory_rw_debug(current_cpu, s->ptr, s->data, s->ptr_len,1);
150 //printf("goldfish_tty_write: read %d bytes to %llx\n", s->ptr_len, (unsigned long long)s->ptr);
151 if(s->data_count > s->ptr_len)
152 memmove(s->data, s->data + s->ptr_len, s->data_count - s->ptr_len);
153 s->data_count -= s->ptr_len;
154 if(s->data_count == 0 && s->ready)
155 goldfish_device_set_irq(&s->dev, 0, 0);
156 break;
157
158 default:
159 cpu_abort (cpu_single_env, "goldfish_tty_write: Bad command %x\n", value);
160 };
161 break;
162
163 case TTY_DATA_PTR:
164 uint64_set_low(&s->ptr, value);
165 break;
166
167 case TTY_DATA_PTR_HIGH:
168 uint64_set_high(&s->ptr, value);
169 break;
170
171 case TTY_DATA_LEN:
172 s->ptr_len = value;
173 break;
174
175 default:
176 cpu_abort (cpu_single_env, "goldfish_tty_write: Bad offset %x\n", offset);
177 }
178 }
179
tty_can_receive(void * opaque)180 static int tty_can_receive(void *opaque)
181 {
182 struct tty_state *s = opaque;
183
184 return (sizeof(s->data) - s->data_count);
185 }
186
tty_receive(void * opaque,const uint8_t * buf,int size)187 static void tty_receive(void *opaque, const uint8_t *buf, int size)
188 {
189 struct tty_state *s = opaque;
190
191 memcpy(s->data + s->data_count, buf, size);
192 s->data_count += size;
193 if(s->data_count > 0 && s->ready)
194 goldfish_device_set_irq(&s->dev, 0, 1);
195 }
196
197 static CPUReadMemoryFunc *goldfish_tty_readfn[] = {
198 goldfish_tty_read,
199 goldfish_tty_read,
200 goldfish_tty_read
201 };
202
203 static CPUWriteMemoryFunc *goldfish_tty_writefn[] = {
204 goldfish_tty_write,
205 goldfish_tty_write,
206 goldfish_tty_write
207 };
208
goldfish_tty_add(CharDriverState * cs,int id,uint32_t base,int irq)209 int goldfish_tty_add(CharDriverState *cs, int id, uint32_t base, int irq)
210 {
211 int ret;
212 struct tty_state *s;
213 static int instance_id = 0;
214
215 s = g_malloc0(sizeof(*s));
216 s->dev.name = "goldfish_tty";
217 s->dev.id = id;
218 s->dev.base = base;
219 s->dev.size = 0x1000;
220 s->dev.irq = irq;
221 s->dev.irq_count = 1;
222 s->cs = cs;
223
224 if(cs) {
225 qemu_chr_add_handlers(cs, tty_can_receive, tty_receive, NULL, s);
226 }
227
228 ret = goldfish_device_add(&s->dev, goldfish_tty_readfn, goldfish_tty_writefn, s);
229 if(ret) {
230 g_free(s);
231 } else {
232 register_savevm(NULL,
233 "goldfish_tty",
234 instance_id++,
235 GOLDFISH_TTY_SAVE_VERSION,
236 goldfish_tty_save,
237 goldfish_tty_load,
238 s);
239 }
240 return ret;
241 }
242
243