Home | History | Annotate | Download | only in hw
      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 "qemu_file.h"
     13 #include "qemu-char.h"
     14 #include "goldfish_device.h"
     15 
     16 #ifdef TARGET_I386
     17 #include "kvm.h"
     18 #endif
     19 
     20 enum {
     21     TTY_PUT_CHAR       = 0x00,
     22     TTY_BYTES_READY    = 0x04,
     23     TTY_CMD            = 0x08,
     24 
     25     TTY_DATA_PTR       = 0x10,
     26     TTY_DATA_LEN       = 0x14,
     27 
     28     TTY_CMD_INT_DISABLE    = 0,
     29     TTY_CMD_INT_ENABLE     = 1,
     30     TTY_CMD_WRITE_BUFFER   = 2,
     31     TTY_CMD_READ_BUFFER    = 3,
     32 };
     33 
     34 struct tty_state {
     35     struct goldfish_device dev;
     36     CharDriverState *cs;
     37     uint32_t ptr;
     38     uint32_t ptr_len;
     39     uint32_t ready;
     40     uint8_t data[128];
     41     uint32_t data_count;
     42 };
     43 
     44 #define  GOLDFISH_TTY_SAVE_VERSION  1
     45 
     46 static void  goldfish_tty_save(QEMUFile*  f, void*  opaque)
     47 {
     48     struct tty_state*  s = opaque;
     49 
     50     qemu_put_be32( f, s->ptr );
     51     qemu_put_be32( f, s->ptr_len );
     52     qemu_put_byte( f, s->ready );
     53     qemu_put_byte( f, s->data_count );
     54     qemu_put_buffer( f, s->data, s->data_count );
     55 }
     56 
     57 static int  goldfish_tty_load(QEMUFile*  f, void*  opaque, int  version_id)
     58 {
     59     struct tty_state*  s = opaque;
     60 
     61     if (version_id != GOLDFISH_TTY_SAVE_VERSION)
     62         return -1;
     63 
     64     s->ptr        = qemu_get_be32(f);
     65     s->ptr_len    = qemu_get_be32(f);
     66     s->ready      = qemu_get_byte(f);
     67     s->data_count = qemu_get_byte(f);
     68     qemu_get_buffer(f, s->data, s->data_count);
     69 
     70     return 0;
     71 }
     72 
     73 static uint32_t goldfish_tty_read(void *opaque, target_phys_addr_t offset)
     74 {
     75     struct tty_state *s = (struct tty_state *)opaque;
     76 
     77     //printf("goldfish_tty_read %x %x\n", offset, size);
     78 
     79     switch (offset) {
     80         case TTY_BYTES_READY:
     81             return s->data_count;
     82     default:
     83         cpu_abort (cpu_single_env, "goldfish_tty_read: Bad offset %x\n", offset);
     84         return 0;
     85     }
     86 }
     87 
     88 static void goldfish_tty_write(void *opaque, target_phys_addr_t offset, uint32_t value)
     89 {
     90     struct tty_state *s = (struct tty_state *)opaque;
     91 
     92     //printf("goldfish_tty_read %x %x %x\n", offset, value, size);
     93 
     94     switch(offset) {
     95         case TTY_PUT_CHAR: {
     96             uint8_t ch = value;
     97             if(s->cs)
     98                 qemu_chr_write(s->cs, &ch, 1);
     99         } break;
    100 
    101         case TTY_CMD:
    102             switch(value) {
    103                 case TTY_CMD_INT_DISABLE:
    104                     if(s->ready) {
    105                         if(s->data_count > 0)
    106                             goldfish_device_set_irq(&s->dev, 0, 0);
    107                         s->ready = 0;
    108                     }
    109                     break;
    110 
    111                 case TTY_CMD_INT_ENABLE:
    112                     if(!s->ready) {
    113                         if(s->data_count > 0)
    114                             goldfish_device_set_irq(&s->dev, 0, 1);
    115                         s->ready = 1;
    116                     }
    117                     break;
    118 
    119                 case TTY_CMD_WRITE_BUFFER:
    120                     if(s->cs) {
    121                         int len;
    122                         target_phys_addr_t  buf;
    123 
    124                         buf = s->ptr;
    125                         len = s->ptr_len;
    126 
    127                         while (len) {
    128                             char   temp[64];
    129                             int    to_write = sizeof(temp);
    130                             if (to_write > len)
    131                                 to_write = len;
    132 
    133 #ifdef TARGET_I386
    134                             if (kvm_enabled())
    135                                 cpu_synchronize_state(cpu_single_env, 0);
    136 #endif
    137                             cpu_memory_rw_debug(cpu_single_env, 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 %x\n", s->ptr_len, 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 #ifdef TARGET_I386
    150                     if (kvm_enabled())
    151                         cpu_synchronize_state(cpu_single_env, 0);
    152 #endif
    153                     cpu_memory_rw_debug(cpu_single_env,s->ptr, s->data, s->ptr_len,1);
    154                     //printf("goldfish_tty_write: read %d bytes to %x\n", s->ptr_len, s->ptr);
    155                     if(s->data_count > s->ptr_len)
    156                         memmove(s->data, s->data + s->ptr_len, s->data_count - s->ptr_len);
    157                     s->data_count -= s->ptr_len;
    158                     if(s->data_count == 0 && s->ready)
    159                         goldfish_device_set_irq(&s->dev, 0, 0);
    160                     break;
    161 
    162                 default:
    163                     cpu_abort (cpu_single_env, "goldfish_tty_write: Bad command %x\n", value);
    164             };
    165             break;
    166 
    167         case TTY_DATA_PTR:
    168             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 
    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 
    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 
    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 = qemu_mallocz(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         qemu_free(s);
    231     } else {
    232         register_savevm( "goldfish_tty", instance_id++, GOLDFISH_TTY_SAVE_VERSION,
    233                          goldfish_tty_save, goldfish_tty_load, s);
    234     }
    235     return ret;
    236 }
    237 
    238