/* Copyright (C) 2007-2008 The Android Open Source Project ** ** This software is licensed under the terms of the GNU General Public ** License version 2, as published by the Free Software Foundation, and ** may be copied, distributed, and modified under those terms. ** ** This program is distributed in the hope that it will be useful, ** but WITHOUT ANY WARRANTY; without even the implied warranty of ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ** GNU General Public License for more details. */ /* implement the modem character device for Android within the QEMU event loop. * it communicates through a serial port with "rild" (Radio Interface Layer Daemon) * on the emulated device. */ #include "modem_driver.h" #include "sysemu/char.h" #define xxDEBUG #ifdef DEBUG # include <stdio.h> # define D(...) ( fprintf( stderr, __VA_ARGS__ ) ) #else # define D(...) ((void)0) #endif AModem android_modem; CharDriverState* android_modem_cs; typedef struct { CharDriverState* cs; AModem modem; char in_buff[ 1024 ]; int in_pos; int in_sms; } ModemDriver; /* send unsollicited messages to the device */ static void modem_driver_unsol( void* _md, const char* message) { ModemDriver* md = _md; int len = strlen(message); qemu_chr_write(md->cs, (const uint8_t*)message, len); } static int modem_driver_can_read( void* _md ) { ModemDriver* md = _md; int ret = sizeof(md->in_buff) - md->in_pos; return ret; } /* despite its name, this function is called when the device writes to the modem */ static void modem_driver_read( void* _md, const uint8_t* src, int len ) { ModemDriver* md = _md; const uint8_t* end = src + len; int nn; D( "%s: reading %d from %p bytes:", __FUNCTION__, len, src ); for (nn = 0; nn < len; nn++) { int c = src[nn]; if (c >= 32 && c < 127) D( "%c", c ); else if (c == '\n') D( "<LF>" ); else if (c == '\r') D( "<CR>" ); else D( "\\x%02x", c ); } D( "\n" ); for ( ; src < end; src++ ) { char c = src[0]; if (md->in_sms) { if (c != 26) goto AppendChar; md->in_buff[ md->in_pos ] = c; md->in_pos++; md->in_sms = 0; c = '\n'; } if (c == '\n' || c == '\r') { const char* answer; if (md->in_pos == 0) /* skip empty lines */ continue; md->in_buff[ md->in_pos ] = 0; md->in_pos = 0; D( "%s: << %s\n", __FUNCTION__, md->in_buff ); answer = amodem_send(android_modem, md->in_buff); if (answer != NULL) { D( "%s: >> %s\n", __FUNCTION__, answer ); len = strlen(answer); if (len == 2 && answer[0] == '>' && answer[1] == ' ') md->in_sms = 1; qemu_chr_write(md->cs, (const uint8_t*)answer, len); qemu_chr_write(md->cs, (const uint8_t*)"\r", 1); } else D( "%s: -- NO ANSWER\n", __FUNCTION__ ); continue; } AppendChar: md->in_buff[ md->in_pos++ ] = c; if (md->in_pos == sizeof(md->in_buff)) { /* input is too long !! */ md->in_pos = 0; } } D( "%s: done\n", __FUNCTION__ ); } static void modem_driver_init( int base_port, ModemDriver* dm, CharDriverState* cs ) { dm->cs = cs; dm->in_pos = 0; dm->in_sms = 0; dm->modem = amodem_create( base_port, modem_driver_unsol, dm ); qemu_chr_add_handlers( cs, modem_driver_can_read, modem_driver_read, NULL, dm ); } void android_modem_init( int base_port ) { static ModemDriver modem_driver[1]; if (android_modem_cs != NULL) { modem_driver_init( base_port, modem_driver, android_modem_cs ); android_modem = modem_driver->modem; } }