
Universal Translator
Dependencies: EthernetNetIf TextLCD mbed PS2 HTTPClient
Revision 0:c69af1faeb95, committed 2011-02-28
- Comitter:
- benglish6
- Date:
- Mon Feb 28 16:56:11 2011 +0000
- Child:
- 1:5ae213418d04
- Commit message:
Changed in this revision
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/EthernetNetIf.lib Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,1 @@ +http://mbed.org/users/donatien/code/EthernetNetIf/#bc7df6da7589
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/FATFileSystem.lib Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,1 @@ +http://mbed.org/users/mbed_unsupported/code/fatfilesystem/ \ No newline at end of file
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/HTTPClient.lib Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,1 @@ +http://mbed.org/users/mamezu/code/HTTPClient/#62fac7f06c8d
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/PS2.lib Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,1 @@ +http://mbed.org/users/shintamainjp/code/PS2/#a57bbbec16b1
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/SDFileSystem.cpp Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,457 @@ +/* mbed SDFileSystem Library, for providing file access to SD cards + * Copyright (c) 2008-2010, sford + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ + +/* Introduction + * ------------ + * SD and MMC cards support a number of interfaces, but common to them all + * is one based on SPI. This is the one I'm implmenting because it means + * it is much more portable even though not so performant, and we already + * have the mbed SPI Interface! + * + * The main reference I'm using is Chapter 7, "SPI Mode" of: + * http://www.sdcard.org/developers/tech/sdcard/pls/Simplified_Physical_Layer_Spec.pdf + * + * SPI Startup + * ----------- + * The SD card powers up in SD mode. The SPI interface mode is selected by + * asserting CS low and sending the reset command (CMD0). The card will + * respond with a (R1) response. + * + * CMD8 is optionally sent to determine the voltage range supported, and + * indirectly determine whether it is a version 1.x SD/non-SD card or + * version 2.x. I'll just ignore this for now. + * + * ACMD41 is repeatedly issued to initialise the card, until "in idle" + * (bit 0) of the R1 response goes to '0', indicating it is initialised. + * + * You should also indicate whether the host supports High Capicity cards, + * and check whether the card is high capacity - i'll also ignore this + * + * SPI Protocol + * ------------ + * The SD SPI protocol is based on transactions made up of 8-bit words, with + * the host starting every bus transaction by asserting the CS signal low. The + * card always responds to commands, data blocks and errors. + * + * The protocol supports a CRC, but by default it is off (except for the + * first reset CMD0, where the CRC can just be pre-calculated, and CMD8) + * I'll leave the CRC off I think! + * + * Standard capacity cards have variable data block sizes, whereas High + * Capacity cards fix the size of data block to 512 bytes. I'll therefore + * just always use the Standard Capacity cards with a block size of 512 bytes. + * This is set with CMD16. + * + * You can read and write single blocks (CMD17, CMD25) or multiple blocks + * (CMD18, CMD25). For simplicity, I'll just use single block accesses. When + * the card gets a read command, it responds with a response token, and then + * a data token or an error. + * + * SPI Command Format + * ------------------ + * Commands are 6-bytes long, containing the command, 32-bit argument, and CRC. + * + * +---------------+------------+------------+-----------+----------+--------------+ + * | 01 | cmd[5:0] | arg[31:24] | arg[23:16] | arg[15:8] | arg[7:0] | crc[6:0] | 1 | + * +---------------+------------+------------+-----------+----------+--------------+ + * + * As I'm not using CRC, I can fix that byte to what is needed for CMD0 (0x95) + * + * All Application Specific commands shall be preceded with APP_CMD (CMD55). + * + * SPI Response Format + * ------------------- + * The main response format (R1) is a status byte (normally zero). Key flags: + * idle - 1 if the card is in an idle state/initialising + * cmd - 1 if an illegal command code was detected + * + * +-------------------------------------------------+ + * R1 | 0 | arg | addr | seq | crc | cmd | erase | idle | + * +-------------------------------------------------+ + * + * R1b is the same, except it is followed by a busy signal (zeros) until + * the first non-zero byte when it is ready again. + * + * Data Response Token + * ------------------- + * Every data block written to the card is acknowledged by a byte + * response token + * + * +----------------------+ + * | xxx | 0 | status | 1 | + * +----------------------+ + * 010 - OK! + * 101 - CRC Error + * 110 - Write Error + * + * Single Block Read and Write + * --------------------------- + * + * Block transfers have a byte header, followed by the data, followed + * by a 16-bit CRC. In our case, the data will always be 512 bytes. + * + * +------+---------+---------+- - - -+---------+-----------+----------+ + * | 0xFE | data[0] | data[1] | | data[n] | crc[15:8] | crc[7:0] | + * +------+---------+---------+- - - -+---------+-----------+----------+ + */ + +#include "SDFileSystem.h" + +#define SD_COMMAND_TIMEOUT 5000 + +SDFileSystem::SDFileSystem(PinName mosi, PinName miso, PinName sclk, PinName cs, const char* name) : + FATFileSystem(name), _spi(mosi, miso, sclk), _cs(cs) { + _cs = 1; +} + +#define R1_IDLE_STATE (1 << 0) +#define R1_ERASE_RESET (1 << 1) +#define R1_ILLEGAL_COMMAND (1 << 2) +#define R1_COM_CRC_ERROR (1 << 3) +#define R1_ERASE_SEQUENCE_ERROR (1 << 4) +#define R1_ADDRESS_ERROR (1 << 5) +#define R1_PARAMETER_ERROR (1 << 6) + +// Types +// - v1.x Standard Capacity +// - v2.x Standard Capacity +// - v2.x High Capacity +// - Not recognised as an SD Card + +#define SDCARD_FAIL 0 +#define SDCARD_V1 1 +#define SDCARD_V2 2 +#define SDCARD_V2HC 3 + +int SDFileSystem::initialise_card() { + // Set to 100kHz for initialisation, and clock card with cs = 1 + _spi.frequency(100000); + _cs = 1; + for(int i=0; i<16; i++) { + _spi.write(0xFF); + } + + // send CMD0, should return with all zeros except IDLE STATE set (bit 0) + if(_cmd(0, 0) != R1_IDLE_STATE) { + fprintf(stderr, "No disk, or could not put SD card in to SPI idle state\n"); + return SDCARD_FAIL; + } + + // send CMD8 to determine whther it is ver 2.x + int r = _cmd8(); + if(r == R1_IDLE_STATE) { + return initialise_card_v2(); + } else if(r == (R1_IDLE_STATE | R1_ILLEGAL_COMMAND)) { + return initialise_card_v1(); + } else { + fprintf(stderr, "Not in idle state after sending CMD8 (not an SD card?)\n"); + return SDCARD_FAIL; + } +} + +int SDFileSystem::initialise_card_v1() { + for(int i=0; i<SD_COMMAND_TIMEOUT; i++) { + _cmd(55, 0); + if(_cmd(41, 0) == 0) { + return SDCARD_V1; + } + } + + fprintf(stderr, "Timeout waiting for v1.x card\n"); + return SDCARD_FAIL; +} + +int SDFileSystem::initialise_card_v2() { + + for(int i=0; i<SD_COMMAND_TIMEOUT; i++) { + _cmd(55, 0); + if(_cmd(41, 0) == 0) { + _cmd58(); + return SDCARD_V2; + } + } + + fprintf(stderr, "Timeout waiting for v2.x card\n"); + return SDCARD_FAIL; +} + +int SDFileSystem::disk_initialize() { + + int i = initialise_card(); +// printf("init card = %d\n", i); +// printf("OK\n"); + + _sectors = _sd_sectors(); + + // Set block length to 512 (CMD16) + if(_cmd(16, 512) != 0) { + fprintf(stderr, "Set 512-byte block timed out\n"); + return 1; + } + + _spi.frequency(1000000); // Set to 1MHz for data transfer + return 0; +} + +int SDFileSystem::disk_write(const char *buffer, int block_number) { + // set write address for single block (CMD24) + if(_cmd(24, block_number * 512) != 0) { + return 1; + } + + // send the data block + _write(buffer, 512); + return 0; +} + +int SDFileSystem::disk_read(char *buffer, int block_number) { + // set read address for single block (CMD17) + if(_cmd(17, block_number * 512) != 0) { + return 1; + } + + // receive the data + _read(buffer, 512); + return 0; +} + +int SDFileSystem::disk_status() { return 0; } +int SDFileSystem::disk_sync() { return 0; } +int SDFileSystem::disk_sectors() { return _sectors; } + +// PRIVATE FUNCTIONS + +int SDFileSystem::_cmd(int cmd, int arg) { + _cs = 0; + + // send a command + _spi.write(0x40 | cmd); + _spi.write(arg >> 24); + _spi.write(arg >> 16); + _spi.write(arg >> 8); + _spi.write(arg >> 0); + _spi.write(0x95); + + // wait for the repsonse (response[7] == 0) + for(int i=0; i<SD_COMMAND_TIMEOUT; i++) { + int response = _spi.write(0xFF); + if(!(response & 0x80)) { + _cs = 1; + _spi.write(0xFF); + return response; + } + } + _cs = 1; + _spi.write(0xFF); + return -1; // timeout +} +int SDFileSystem::_cmdx(int cmd, int arg) { + _cs = 0; + + // send a command + _spi.write(0x40 | cmd); + _spi.write(arg >> 24); + _spi.write(arg >> 16); + _spi.write(arg >> 8); + _spi.write(arg >> 0); + _spi.write(0x95); + + // wait for the repsonse (response[7] == 0) + for(int i=0; i<SD_COMMAND_TIMEOUT; i++) { + int response = _spi.write(0xFF); + if(!(response & 0x80)) { + return response; + } + } + _cs = 1; + _spi.write(0xFF); + return -1; // timeout +} + + +int SDFileSystem::_cmd58() { + _cs = 0; + int arg = 0; + + // send a command + _spi.write(0x40 | 58); + _spi.write(arg >> 24); + _spi.write(arg >> 16); + _spi.write(arg >> 8); + _spi.write(arg >> 0); + _spi.write(0x95); + + // wait for the repsonse (response[7] == 0) + for(int i=0; i<SD_COMMAND_TIMEOUT; i++) { + int response = _spi.write(0xFF); + if(!(response & 0x80)) { + int ocr = _spi.write(0xFF) << 24; + ocr |= _spi.write(0xFF) << 16; + ocr |= _spi.write(0xFF) << 8; + ocr |= _spi.write(0xFF) << 0; +// printf("OCR = 0x%08X\n", ocr); + _cs = 1; + _spi.write(0xFF); + return response; + } + } + _cs = 1; + _spi.write(0xFF); + return -1; // timeout +} + +int SDFileSystem::_cmd8() { + _cs = 0; + + // send a command + _spi.write(0x40 | 8); // CMD8 + _spi.write(0x00); // reserved + _spi.write(0x00); // reserved + _spi.write(0x01); // 3.3v + _spi.write(0xAA); // check pattern + _spi.write(0x87); // crc + + // wait for the repsonse (response[7] == 0) + for(int i=0; i<SD_COMMAND_TIMEOUT * 1000; i++) { + char response[5]; + response[0] = _spi.write(0xFF); + if(!(response[0] & 0x80)) { + for(int j=1; j<5; j++) { + response[i] = _spi.write(0xFF); + } + _cs = 1; + _spi.write(0xFF); + return response[0]; + } + } + _cs = 1; + _spi.write(0xFF); + return -1; // timeout +} + +int SDFileSystem::_read(char *buffer, int length) { + _cs = 0; + + // read until start byte (0xFF) + while(_spi.write(0xFF) != 0xFE); + + // read data + for(int i=0; i<length; i++) { + buffer[i] = _spi.write(0xFF); + } + _spi.write(0xFF); // checksum + _spi.write(0xFF); + + _cs = 1; + _spi.write(0xFF); + return 0; +} + +int SDFileSystem::_write(const char *buffer, int length) { + _cs = 0; + + // indicate start of block + _spi.write(0xFE); + + // write the data + for(int i=0; i<length; i++) { + _spi.write(buffer[i]); + } + + // write the checksum + _spi.write(0xFF); + _spi.write(0xFF); + + // check the repsonse token + if((_spi.write(0xFF) & 0x1F) != 0x05) { + _cs = 1; + _spi.write(0xFF); + return 1; + } + + // wait for write to finish + while(_spi.write(0xFF) == 0); + + _cs = 1; + _spi.write(0xFF); + return 0; +} + +static int ext_bits(char *data, int msb, int lsb) { + int bits = 0; + int size = 1 + msb - lsb; + for(int i=0; i<size; i++) { + int position = lsb + i; + int byte = 15 - (position >> 3); + int bit = position & 0x7; + int value = (data[byte] >> bit) & 1; + bits |= value << i; + } + return bits; +} + +int SDFileSystem::_sd_sectors() { + + // CMD9, Response R2 (R1 byte + 16-byte block read) + if(_cmdx(9, 0) != 0) { + fprintf(stderr, "Didn't get a response from the disk\n"); + return 0; + } + + char csd[16]; + if(_read(csd, 16) != 0) { + fprintf(stderr, "Couldn't read csd response from disk\n"); + return 0; + } + + // csd_structure : csd[127:126] + // c_size : csd[73:62] + // c_size_mult : csd[49:47] + // read_bl_len : csd[83:80] - the *maximum* read block length + + int csd_structure = ext_bits(csd, 127, 126); + int c_size = ext_bits(csd, 73, 62); + int c_size_mult = ext_bits(csd, 49, 47); + int read_bl_len = ext_bits(csd, 83, 80); + +// printf("CSD_STRUCT = %d\n", csd_structure); + + if(csd_structure != 0) { + fprintf(stderr, "This disk tastes funny! I only know about type 0 CSD structures\n"); + return 0; + } + + // memory capacity = BLOCKNR * BLOCK_LEN + // where + // BLOCKNR = (C_SIZE+1) * MULT + // MULT = 2^(C_SIZE_MULT+2) (C_SIZE_MULT < 8) + // BLOCK_LEN = 2^READ_BL_LEN, (READ_BL_LEN < 12) + + int block_len = 1 << read_bl_len; + int mult = 1 << (c_size_mult + 2); + int blocknr = (c_size + 1) * mult; + int capacity = blocknr * block_len; + + int blocks = capacity / 512; + + return blocks; +}
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/SDFileSystem.h Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,81 @@ +/* mbed SDFileSystem Library, for providing file access to SD cards + * Copyright (c) 2008-2010, sford + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ + +#ifndef MBED_SDFILESYSTEM_H +#define MBED_SDFILESYSTEM_H + +#include "mbed.h" +#include "FATFileSystem.h" + +/** Access the filesystem on an SD Card using SPI + * + * @code + * #include "mbed.h" + * #include "SDFileSystem.h" + * + * SDFileSystem sd(p5, p6, p7, p12, "sd"); // mosi, miso, sclk, cs + * + * int main() { + * FILE *fp = fopen("/sd/myfile.txt", "w"); + * fprintf(fp, "Hello World!\n"); + * fclose(fp); + * } + */ +class SDFileSystem : public FATFileSystem { +public: + + /** Create the File System for accessing an SD Card using SPI + * + * @param mosi SPI mosi pin connected to SD Card + * @param miso SPI miso pin conencted to SD Card + * @param sclk SPI sclk pin connected to SD Card + * @param cs DigitalOut pin used as SD Card chip select + * @param name The name used to access the virtual filesystem + */ + SDFileSystem(PinName mosi, PinName miso, PinName sclk, PinName cs, const char* name); + virtual int disk_initialize(); + virtual int disk_write(const char *buffer, int block_number); + virtual int disk_read(char *buffer, int block_number); + virtual int disk_status(); + virtual int disk_sync(); + virtual int disk_sectors(); + +protected: + + int _cmd(int cmd, int arg); + int _cmdx(int cmd, int arg); + int _cmd8(); + int _cmd58(); + int initialise_card(); + int initialise_card_v1(); + int initialise_card_v2(); + + int _read(char *buffer, int length); + int _write(const char *buffer, int length); + int _sd_sectors(); + int _sectors; + + SPI _spi; + DigitalOut _cs; +}; + +#endif
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/TextLCD.lib Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,1 @@ +http://mbed.org/users/simon/code/TextLCD/#44f34c09bd37
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/UnivTrans.h Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,34 @@ +#include "TextLCD.h" +#include "EthernetNetIf.h" +#include "HTTPClient.h" +#include "PS2Keyboard.h" +#include "SDFileSystem.h" +#include "VS1002.h" +#include <string> + +enum codeenum { +af,it,sq,ja,ar,ko,be,lv,bg,lt,ca,mk,zh,ms,zhCN,mt,zhTW,no,hr,fa,cs,pl,da,pt,nl,ptPT,en,ro,et,ru,tl,sr,fi,sk, +fr,sl,gl,es,de,sw,el,sv,ht,tgl,iw,th,hi,tr,hu,uk,is,vi,id,cy,ga,yi,nolang +}; + +struct tcode { + int src; + int dest; +}; + +extern TextLCD lcd; +extern DigitalOut led1; +extern DigitalOut led2; +extern DigitalOut led3; +extern DigitalOut led4; + +extern EthernetNetIf eth; // Ethernet Control +extern HTTPClient http; // HTTP Client Stack Library + +extern VS1002 mp3; + +string getLang(int); +string getLangCode(int); +void encodeURL(string&, const char*); +void blink(void); +void checkKeys(PS2Keyboard&,char*); \ No newline at end of file
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/VS1002.cpp Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,286 @@ +#include "VS1002.h" +#include "mbed.h" +#include "TextLCD.h" +//AnalogIn rotary1(p19); +//DigitalIn pause1(p28); +//TextLCD lcd(p10, p18, p24, p23, p22, p21, TextLCD::LCD16x2 ); // rs, e, d0-d3 + + +/* ================================================================== + * Constructor + * =================================================================*/ +VS1002::VS1002( +PinName mmosi, PinName mmiso, PinName ssck, PinName ccs, const char *name, + PinName mosi, PinName miso, PinName sck, PinName cs, PinName rst, + PinName dreq, PinName dcs, PinName vol) + : + _sd(mmosi, mmiso, ssck, ccs, name), + _spi(mosi, miso, sck), + _CS(cs), + _RST(rst), + _DREQ(dreq), + _DCS(dcs), + _VOL(vol) { + + } + +/*=================================================================== + * Functions + *==================================================================*/ + +void VS1002::cs_low(void) +{ + _CS = 0; +} +void VS1002::cs_high(void) +{ + _CS = 1; +} +void VS1002::dcs_low(void) +{ + _DCS = 0; +} +void VS1002::dcs_high(void) +{ + _DCS = 1; +} +void VS1002::sci_en(void) //SCI enable +{ + cs_high(); + dcs_high(); + cs_low(); +} +void VS1002::sci_dis(void) //SCI disable +{ + cs_high(); +} +void VS1002::sdi_en(void) //SDI enable +{ + dcs_high(); + cs_high(); + dcs_low(); +} +void VS1002::sdi_dis(void) //SDI disable +{ + dcs_high(); +} +void VS1002::reset(void) //hardware reset +{ + wait(0.01); + _RST = 0; + wait(0.01); + _RST = 1; + wait(0.10); +} +void VS1002::power_down(void) //hardware and software reset +{ + cs_low(); + reset(); + sci_write(0x00, SM_PDOWN); + wait(0.01); + reset(); +} +void VS1002::sci_initialise(void) +{ + _RST = 1; //no reset + _spi.format(8,0); //spi 8bit interface, steady state low + _spi.frequency(1000000); //rising edge data record, freq. 1Mhz + + cs_low(); + for(int i=0; i<4; i++) + { + _spi.write(0xFF); //clock the chip a bit + } + cs_high(); + dcs_high(); + wait_us(5); +} +void VS1002::sdi_initialise(void) +{ + _spi.format(8,0); + _spi.frequency(7000000); //set to 7MHz + + cs_high(); + dcs_high(); +} +void VS1002::sci_write(unsigned char address, unsigned short int data) +{ + sci_en(); //enables SCI/disables SDI + + while(!_DREQ); //wait unitl data request is high + _spi.write(0x02); //SCI write + _spi.write(address); //register address + _spi.write((data >> 8) & 0xFF); //write out first half of data word + _spi.write(data & 0xFF); //write out second half of data word + + sci_dis(); //enables SDI/disables SCI + wait_us(5); +} +void VS1002::sdi_write(unsigned char datum) +{ + sdi_en(); + + while(!_DREQ); + _spi.write(datum); + + sci_dis(); +} +unsigned short int VS1002::read(unsigned short int address) +{ + cs_low(); //enables SCI/disables SDI + + while(!_DREQ); //wait unitl data request is high + _spi.write(0x03); //SCI write + _spi.write(address); //register address + unsigned short int received = _spi.write(0x00); //write out dummy byte + received <<= 8; + received += _spi.write(0x00); //write out dummy byte + + cs_high(); //enables SDI/disables SCI + + return received; //return received word +} +void VS1002::sine_test_activate(unsigned char wave) +{ + cs_high(); //enables SDI/disables SCI + + while(!_DREQ); //wait unitl data request is high + _spi.write(0x53); //SDI write + _spi.write(0xEF); //SDI write + _spi.write(0x6E); //SDI write + _spi.write(wave); //SDI write + _spi.write(0x00); //filler byte + _spi.write(0x00); //filler byte + _spi.write(0x00); //filler byte + _spi.write(0x00); //filler byte + + cs_low(); //enables SCI/disables SDI +} +void VS1002::sine_test_deactivate(void) +{ + cs_high(); + + while(!_DREQ); + _spi.write(0x45); //SDI write + _spi.write(0x78); //SDI write + _spi.write(0x69); //SDI write + _spi.write(0x74); //SDI write + _spi.write(0x00); //filler byte + _spi.write(0x00); //filler byte + _spi.write(0x00); //filler byte + _spi.write(0x00); //filler byte +} +void VS1002::volume(void) +{ + #ifdef FIXED_VOL + unsigned char volumize = (0 * 255); // FIXED VOL (not support volume input) + #else + unsigned char volumize = (_VOL * 255); + #endif + while(!_DREQ); + + unsigned short int attenuation = ((256 * volumize) + volumize); + sci_write(0x0B, attenuation); + +} + +void VS1002::play_song(int song_number) +{ + /*====== Song Select ======*/ + +// char list[10000] = {0}; + char list[1000] = {0}; + char str[16] = {"/sd/"}; + unsigned int startplace = 0; + unsigned int endplace = 0; + unsigned int play = 0; + num_of_files = 0; + + DIR *d; + struct dirent *p; + d = opendir("/sd"); + if(d != NULL) + { + while((p = readdir(d)) != NULL) + { + strcat(list, "*"); + strcat(list, p->d_name); + num_of_files++; + printf("hi!%i\n",num_of_files); + } + } + else + { + perror("Could not open directory!"); + } + strcat(list, "*"); //terminating * + if(num_of_files < song_number) + { + return; + } + while(play != song_number) + { + char symbol = list[startplace]; + startplace++; + if(symbol == 0x2A) //0x2A = "*" + { + play++; + } + } + play = 0; + while(play != (song_number+1)) + { + char symbol = list[endplace]; + endplace++; + if(symbol == 0x2A) //0x2A = "*" + { + play++; + } + } + + strncat(str, &list[startplace], endplace-startplace); + str[(endplace-startplace)+3] = '\0'; + +//printf("list: %s\r\n",list); //debug + + /*====== File Transfer ======*/ + + // return if not MP3 file + if (!((strstr(str,"MP3")!=NULL)||(strstr(str,"mp3")!=NULL))) return; + // display filename.mp3 to the lcd screen + //lcd.printf("Now Playing: %s\r\n",str); + + FILE *song; + unsigned char array[512]; + + song = fopen(str, "rb"); + + if(!song) + { + printf("oh noes! no song!\n"); + exit(1); + } + + while((!feof(song))/*&&!pause1*/) + { + printf("should be playing\n"); + fread(&array, 1, 512, song); + for(int i=0; i<512; i++) + { +#ifndef FS_ONLY + sdi_write(array[i]); + // printf("."); +#endif + } +#ifndef FS_ONLY + volume(); +#endif + } + for(int n=0; n<2048; n++) + { +#ifndef FS_ONLY + sdi_write(0x00); +#endif + } + fclose(song); //close the file +}
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/VS1002.h Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,74 @@ +#ifndef VS1002_H +#define VS1002_H + +//#define FS_ONLY +//#define FIXED_VOL + +#include "mbed.h" +#include "SDFileSystem.h" +#include "string" +#include "string.h" + +//SCI_MODE register bits as of p.26 of the datasheet +#define SM_DIFF 0x0001 +#define SM_SETTOZERO 0x0002 +#define SM_RESET 0x0004 +#define SM_OUTOFWAV 0x0008 +#define SM_PDOWN 0x0010 +#define SM_TESTS 0x0020 +#define SM_STREAM 0x0040 +#define SM_PLUSV 0x0080 +#define SM_DACT 0x0100 +#define SM_SDIORD 0x0200 +#define SM_SDISHARE 0x0400 +#define SM_SDINEW 0x0800 +#define SM_ADPCM 0x1000 +#define SM_ADPCM_HP 0x2000 + + +class VS1002 { + +public: +// VS1002(int _mmosi, int _mmiso, int _ssck, int _ccs, const char* _name, +// int _mosi, int _miso, int _sck, int _cs, int _rst, int _dreq, int _dcs, int _vol); + VS1002(PinName _mmosi, PinName _mmiso, PinName _ssck, PinName _ccs, const char* _name, + PinName _mosi, PinName _miso, PinName _sck, PinName _cs, PinName _rst, PinName _dreq, + PinName _dcs, PinName _vol); + + void cs_low(void); + void cs_high(void); + void dcs_low(void); + void dcs_high(void); + void sci_en(void); + void sci_dis(void); + void sdi_en(void); + void sdi_dis(void); + + void sci_initialise(void); + void sdi_initialise(void); + void reset(void); + void power_down(void); + + void sci_write(unsigned char, unsigned short int); + void sdi_write(unsigned char); + unsigned short int read(unsigned short int); + void sine_test_activate(unsigned char); + void volume(void); + void sine_test_deactivate(void); + void play_song(int); + + int num_of_files; + + DigitalIn _DREQ; + DigitalOut _RST; + AnalogIn _VOL; + +protected: + + SPI _spi; + DigitalOut _CS; + DigitalOut _DCS; + SDFileSystem _sd; + +}; +#endif
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/keyProcess.cpp Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,242 @@ +#include "UnivTrans.h" + +static const unsigned char ps2KeyMap[] = { +// 0 1 2 3 4 5 6 7 8 9 A B C D E F + '\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0', '`','\0', // 00-0F + '\0','\0','\0','\0','\0', 'Q', '1','\0','\0','\0', 'Z', 'S', 'A', 'W', '2','\0', // 10-1F + '\0', 'C', 'X', 'D', 'E', '4', '3','\0','\0',' ', 'V', 'F', 'T', 'R', '5','\0', // 20-2F + '\0', 'N', 'B', 'H', 'G', 'Y', '6','\0','\0', '\0', 'M', 'J', 'U', '7', '8','\0', // 30-3F + '\0', ',', 'K', 'I', 'O', '0', '9','\0','\0', '.', '/', 'L', ';', 'P', '-','\0', // 40-4F + '\0','\0', '\'','\0', '[', '=','\0','\0','\0','\0','\0', ']','\0','\\','\0','\0', // 50-5F + '\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0', // 60-6F + '\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0', // 70-7F + '\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0', // 80-8F + '\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0', // 90-9F + '\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0', // A0-AF + '\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0', // B0-BF + '\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0', // C0-CF + '\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0', // D0-DF + '\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0', // E0-EF + '\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0','\0', // F0-FF +}; + +void checkKeys(PS2Keyboard &ps2kb,char* rowstr) { + static int column = 0; + static int row = 0; + static int selected = true; + static int source = false; + static const char *selsrc="Select Src Lang\n"; + static const char *seldest="Select Dest Lang"; + static string httpurl = "http://translate.google.com/translate_tts?tl="; + static string httptransurl = " ";//INSERT YOUR WEBSITE HERE! + static string arg = "&q="; + string voiceurl; + string transurl; + static int src=26; + static int dest=37; + string lang; + string translatedText; + static PS2Keyboard::keyboard_event_t evt_kb; +if (ps2kb.processing(&evt_kb)) { + for (int i = 0; i < evt_kb.length; i++) { + if(evt_kb.type == 1) { + if(!selected) { + if (evt_kb.scancode[i] == 0x72) {//down arrow + if (source) src++; + else dest++; + lang = getLang(source?src:dest); + } else if (evt_kb.scancode[i] == 0x75) { //up arrow + if (source && src!=0) src--; + else if (!source && dest!=0) dest--; + lang = getLang(source?src:dest); + } else if(evt_kb.scancode[i] == 0x5A) { + if(source) { + source = false; + lang = getLang(dest); + } else selected = true; + } + if(selected) { + lcd.cls(); + voiceurl += httpurl; + voiceurl += getLangCode(dest); + voiceurl += arg; + transurl += httptransurl; + transurl += getLangCode(src); + transurl += "&dest="; + transurl += getLangCode(dest); + transurl += arg; + encodeURL(transurl,rowstr); + printf("%s",transurl.c_str()); + HTTPText t; + HTTPResult r = http.get(transurl.c_str(),&t); + if(r==HTTP_OK) { + printf("Result OK\n"); + } else { + printf("Error %d\n", r); + } + translatedText=t.get(); + encodeURL(voiceurl,translatedText.c_str()); + printf("%s",voiceurl.c_str()); + HTTPFile f("/sd/alaha.mp3"); + r = http.get(voiceurl.c_str(), &f); + if(r==HTTP_OK) { + printf("Result OK\n"); + } else { + printf("Error %d\n", r); + } + mp3.play_song(1); + voiceurl.erase(); + transurl.erase(); + memset(rowstr,0,6*16); + } else { + lcd.cls(); + lcd.printf(source?selsrc:seldest); + lcd.printf(lang.c_str()); + } + } else { + if (ps2KeyMap[evt_kb.scancode[i]] != '\0'){ + if(row == 0){ + rowstr[(row*16)+column]=ps2KeyMap[evt_kb.scancode[i]]; + if(column > 14) { + column = 0; + row++; + if(row > 6) row = 0; + } + else column++; + lcd.cls(); + for(int j = 0; j < 32; j++) { + if(rowstr[(row*16)+j] != '\0') lcd.printf("%c", rowstr[(row*16)+j]); + if((j+1) == column) lcd.printf("%c", '\0'); + } + } else { + rowstr[(row*16)+column]=ps2KeyMap[evt_kb.scancode[i]]; + if(column > 14) { + column = 0; + row++; + if(row > 6) row = 0; + } + else column++; + lcd.cls(); + for(int j = 0; j < 32; j++) { + if(rowstr[((row-1)*16)+j] != '\0') lcd.printf("%c", rowstr[((row-1)*16)+j]); + if((j+1) == (column+16)) lcd.printf("%c", '\0'); + } + } + } + if (evt_kb.scancode[i] == 0x66) {//backspace deletes a character + if(column > 0) column--; + else { + if (row != 0) { + column = 15; + row--; + } + } + rowstr[(row*16)+column] = '\0'; + lcd.cls(); + if (row != 0) { + for(int j = 0; j < 32; j++) { + if(rowstr[((row-1)*16)+j] != '\0') lcd.printf("%c", rowstr[(row-1)*16+j]); + if(j == (column+16)) lcd.printf("%c", '\0'); + } + } else { + for(int j = 0; j < 32; j++) { + if(rowstr[(row*16)+j] != '\0') lcd.printf("%c", rowstr[(row)*j]); + if(j == column) lcd.printf("%c", '\0'); + } + } + } + if (evt_kb.scancode[i] == 0x5A) {//enter sends the message + selected = false; + source = true; + lang = getLang(src); + lcd.cls(); + lcd.printf(selsrc); + lcd.printf(lang.c_str()); + row = 0; + column = 0; + return; + } + if (evt_kb.scancode[i] == 0x75) {//up arrow + if(row > 1) { + lcd.cls(); + row = row - 1; + for(int j = 0; j < 32; j++) { + if(rowstr[((row-1)*16)+j] != '\0') lcd.printf("%c", rowstr[((row-1)*16)+j]); + } + lcd.locate(column, 1); + lcd.printf("%c", '\0'); + } else { + if (row == 1) { + row = row - 1; + lcd.locate(column, row); + lcd.printf("%c", '\0'); + } + } + + } + if (evt_kb.scancode[i] == 0x72) {//down arrow + if(rowstr[((row+1))+column] != '\0'){ + lcd.cls(); + for(int j = 0; j < 32; j++) { + if(rowstr[(row*16)+j] != '\0') lcd.printf("%c", rowstr[(row*16)+j]); + } + row = row + 1; + lcd.locate(column, 1); + lcd.printf("%c", '\0'); + } + } + if (evt_kb.scancode[i] == 0x6B) {//left arrow + if((column !=0)) { + if(row == 0){ + column = column - 1; + lcd.cls(); + for(int j = 0; j < 32; j++) { + if(rowstr[(row*16)+j] != '\0') lcd.printf("%c", rowstr[(row*16)+j]); + } + lcd.locate(column, 0); + lcd.printf("%c", '\0'); + } else { + column = column - 1; + lcd.cls(); + for(int j = 0; j < 32; j++) { + if(rowstr[((row-1)*16)+j] != '\0') lcd.printf("%c", rowstr[((row-1)*16)+j]); + } + lcd.locate(column, 1); + lcd.printf("%c", '\0'); + } + } + } + if (evt_kb.scancode[i] == 0x74) {//right arrow + if(rowstr[(row*16)+column+1] != '\0'){ + if(column != 15){ + column = column + 1; + if (row == 0){ + lcd.cls(); + for(int j = 0; j < 32; j++) { + if(rowstr[(row*16)+j] != '\0') lcd.printf("%c", rowstr[(row*16)+j]); + } + lcd.locate(column, 0); + lcd.printf("%c", '\0'); + } else { + lcd.cls(); + for(int j = 0; j < 32; j++) { + if(rowstr[((row-1)*16)+j] != '\0') lcd.printf("%c", rowstr[((row-1)*16)+j]); + } + lcd.locate(column, 1); + lcd.printf("%c", '\0'); + } + } + } + } + + if (evt_kb.scancode[i] == 0x71) {//delete key will clear everything + memset(rowstr,0,6*16); + lcd.cls(); + row = 0; + column = 0; + } + } + } + } + } +}
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/languages.cpp Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,364 @@ +#include "UnivTrans.h" + + +/*string codes[] = { +"af","it","sq","ja","ar","ko","be","lv","bg","lt","ca","mk","zh","ms","zhCN","mt","zhTW","no","hr", +"fa","cs","pl","da","pt","nl","ptPT","en","ro","et","ru","tl","sr","fi","sk","fr","sl","gl","es","de", +"sw","el","sv","ht","tgl","iw","th","hi","tr","hu","uk","is","vi","id","cy","ga","yi" +};*/ + +string getLang(int lcode) { + string langstr; + switch (lcode) { + case af: + langstr="Afrikaans"; + break; + case it: + langstr="Italian"; + break; + case sq: + langstr="Albanian"; + break; + case ja: + langstr="JAPANESE"; + break; + case ar: + langstr="ARABIC"; + break; + case ko: + langstr="KOREAN"; + break; + case be: + langstr="BELARUSIAN"; + break; + case lv: + langstr="LATVIAN"; + break; + case bg: + langstr="BULGARIAN"; + break; + case lt: + langstr="LITHUANIAN"; + break; + case ca: + langstr="CATALAN"; + break; + case mk: + langstr="MACEDONIAN"; + break; + case zh: + langstr="CHINESE"; + break; + case ms: + langstr="MALAY "; + break; + case zhCN: + langstr="CHINESE_SIMPLIFIED"; + break; + case mt: + langstr="MALTESE"; + break; + case zhTW: + langstr="CHINESE_TRADITIONAL"; + break; + case no: + langstr="NORWEGIAN"; + break; + case hr: + langstr="CROATIAN"; + break; + case fa: + langstr="PERSIAN"; + break; + case cs: + langstr="CZECH"; + break; + case pl: + langstr="POLISH "; + break; + case da: + langstr="DANISH "; + break; + case pt: + langstr="PORTUGUESE"; + break; + case nl: + langstr="DUTCH"; + break; + case ptPT: + langstr="PORTUGUESE_PORTUGAL"; + break; + case en: + langstr="ENGLISH"; + break; + case ro: + langstr="ROMANIAN"; + break; + case et: + langstr="ESTONIAN"; + break; + case ru: + langstr="RUSSIAN"; + break; + case tl: + langstr="FILIPINO"; + break; + case sr: + langstr="SERBIAN"; + break; + case fi: + langstr="FINNISH"; + break; + case sk: + langstr="SLOVAK"; + break; + case fr: + langstr="FRENCH"; + break; + case sl: + langstr="SLOVENIAN"; + break; + case gl: + langstr="GALICIAN"; + break; + case es: + langstr="SPANISH"; + break; + case de: + langstr="GERMAN"; + break; + case sw: + langstr="SWAHILI"; + break; + case el: + langstr="GREEK"; + break; + case sv: + langstr="SWEDISH"; + break; + case ht: + langstr="HAITIAN_CREOLE"; + break; + case tgl: + langstr="TAGALOG"; + break; + case iw: + langstr="HEBREW"; + break; + case th: + langstr="THAI"; + break; + case hi : + langstr="HINDI"; + break; + case tr: + langstr="TURKISH"; + break; + case hu: + langstr="HUNGARIAN"; + break; + case uk: + langstr="UKRAINIAN"; + break; + case is: + langstr="ICELANDIC"; + break; + case vi: + langstr="VIETNAMESE"; + break; + case id: + langstr="INDONESIAN"; + break; + case cy: + langstr="WELSH"; + break; + case ga: + langstr="IRISH"; + break; + case yi: + langstr="YIDDISH"; + break; + default: + langstr="English"; + break; + } + return langstr; +} + +string getLangCode(int lcode) { + string langcode; + switch (lcode) { + case af: + langcode="af"; + break; + case it: + langcode="it"; + break; + case sq: + langcode="sq"; + break; + case ja: + langcode="ja"; + break; + case ar: + langcode="ar"; + break; + case ko: + langcode="ko"; + break; + case be: + langcode="be"; + break; + case lv: + langcode="lv"; + break; + case bg: + langcode="bg"; + break; + case lt: + langcode="lt"; + break; + case ca: + langcode="ca"; + break; + case mk: + langcode="mk"; + break; + case zh: + langcode="zh"; + break; + case ms: + langcode="ms"; + break; + case zhCN: + langcode="zh-CN"; + break; + case mt: + langcode="mt"; + break; + case zhTW: + langcode="zh-TW"; + break; + case no: + langcode="no"; + break; + case hr: + langcode="hr"; + break; + case fa: + langcode="fa"; + break; + case cs: + langcode="cs"; + break; + case pl: + langcode="pl"; + break; + case da: + langcode="da"; + break; + case pt: + langcode="pt"; + break; + case nl: + langcode="nl"; + break; + case ptPT: + langcode="pt-PT"; + break; + case en: + langcode="en"; + break; + case ro: + langcode="ro"; + break; + case et: + langcode="et"; + break; + case ru: + langcode="ru"; + break; + case tl: + langcode="tl"; + break; + case sr: + langcode="sr"; + break; + case fi: + langcode="fi"; + break; + case sk: + langcode="sk"; + break; + case fr: + langcode="fr"; + break; + case sl: + langcode="sl"; + break; + case gl: + langcode="gl"; + break; + case es: + langcode="es"; + break; + case de: + langcode="de"; + break; + case sw: + langcode="sw"; + break; + case el: + langcode="el"; + break; + case sv: + langcode="sv"; + break; + case ht: + langcode="ht"; + break; + case tgl: + langcode="tl"; + break; + case iw: + langcode="iw"; + break; + case th: + langcode="th"; + break; + case hi: + langcode="hi"; + break; + case tr: + langcode="tr"; + break; + case hu: + langcode="hu"; + break; + case uk: + langcode="uk"; + break; + case is: + langcode="is"; + break; + case vi: + langcode="vi"; + break; + case id: + langcode="id"; + break; + case cy: + langcode="cy"; + break; + case ga: + langcode="ga"; + break; + case yi: + langcode="yi"; + break; + default: + langcode="en"; + break; + } + return langcode; +} \ No newline at end of file
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/main.cpp Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,58 @@ +#include "mbed.h" +#include "UnivTrans.h" + +TextLCD lcd(p14, p16, p17, p18, p19, p20); // rs, e, d4-d7 +DigitalOut led1(LED1); +DigitalOut led2(LED2); +DigitalOut led3(LED3); +DigitalOut led4(LED4); + +EthernetNetIf eth; // Ethernet Control +HTTPClient http; // HTTP Client Stack Library + +VS1002 mp3( // Initialize MP3 Hardware Decoder class + p5, p6, p7, p8, "sd", // SD Card SPI Connections & Setup + p11, p12, p13, p10, // VS1053/1002 SPI Connections + p21, p22, p23, p15); // DREQ,DCS,VOL + // Due to the current TextLCD setup, the only Analog pin + // available is p15, thus it is used for volume control + +int main() { + Serial pc(USBTX, USBRX); + PS2Keyboard KeyB(p29, p30); // CLK, DAT + Ticker tm; // Controls Blinking of LEDs/consistent ops + + // Start Setting Up Hardware + printf("Setting up...\n"); + EthernetErr ethErr = eth.setup(); + if(ethErr) { + printf("Error %d in setup.\n", ethErr); + return -1; + } + printf("Setup OK\n"); + + // Initialize MP3 Hardware + #ifndef FS_ONLY + mp3._RST = 1; + mp3.cs_high(); //chip disabled + mp3.sci_initialise(); //initialise MBED + mp3.sci_write(0x00,(SM_SDINEW+SM_STREAM+SM_DIFF)); + mp3.sci_write(0x03, 0x9800); + mp3.sdi_initialise(); + #endif + + //Blink now to signal successful hardware/internet setup + tm.attach(&blink,.5); + + //Now let's get to the real code + + char rowstring[6*16]; + memset(rowstring,0,6*16); + + while(true){ + checkKeys(KeyB,rowstring); + Net::poll(); + } + + return 0; +} \ No newline at end of file
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/mbed.bld Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,1 @@ +http://mbed.org/users/mbed_official/code/mbed/builds/63bcd7ba4912
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/utilities.cpp Mon Feb 28 16:56:11 2011 +0000 @@ -0,0 +1,48 @@ +#include "UnivTrans.h" + +void blink(){ //BlinkAlive function which is attached to ticker after successful setup + led1=!led1; +} + +void encodeURL(string &dest, const char *src) { + for(int i=0;(src[i]!=0)&&(i<6*16);i++) { + switch (src[i]) { + case '$': + dest+="%24"; + break; + case '&': + dest+="%26"; + break; + case '+': + dest+="%2B"; + break; + case ',': + dest+="%2C"; + break; + case '/': + dest+="%2F"; + break; + case ':': + dest+="%3A"; + break; + case ';': + dest+="%3B"; + break; + case '=': + dest+="%3D"; + break; + case '?': + dest+="%3F"; + break; + case '@': + dest+="%40"; + break; + case ' ': + dest+="%20"; + break; + default: + dest+=src[i]; + break; + } + } +} \ No newline at end of file