The iPod controller that I submitted for the mbed challenge

Dependencies:   mbed Motordriver PID

Revision:
0:371773dd3dd1
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/user_interface/TextLCD23017.cpp	Wed May 04 15:41:13 2011 +0000
@@ -0,0 +1,235 @@
+/* mbed TextLCD23017 Library, for a 4-bit LCD based on HD44780
+ * Copyright (c) 2007-2010, sford, http://mbed.org
+ *
+ * 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.
+ */
+#include <stdarg.h>
+#include "TextLCD23017.h"
+#include "mbed.h"
+
+/**************************************************************
+Port A: databus 8-bit bidir, also used as input for keys (A2-A7)
+IntA wired to mbed pin??
+PortB: B0 out: keys out, set low to read keys on PortA
+       B1 in:  rot.enc. push button act.L
+       B2 out: buzzer
+       B3 in:  rot.enc. A
+       B4 in:  rot.enc. B
+       B5 out: LCD E (CS\)
+       B6 out: LCD R\W (WR\)
+       B7 out: LCD RS (A0)
+**************************************************************/
+TextLCD23017::TextLCD23017(MCP23017& intf, LCDType type) : _intf(intf), _type(type) {
+    ::printf("creating lcd\n");
+    wait(0.015);        // Wait 15ms to ensure powered up of LCD
+//init the MCP23017
+    intf.direction(PORT_A, PORT_DIR_IN); //input
+    intf.direction(PORT_B, INPUTS);// rot enc input
+    intf.configurePullUps(PORT_A, PORT_DIR_IN);
+    intf.configurePullUps(PORT_B, INPUTS);
+    intf.write(PORT_B, 0x00); //enable keys
+    //intf.interruptPolarity(ACTIVE_LOW);
+    intf.mirrorInterrupts(true);
+    
+//initialisation according to HD44780 datasheet
+    // send "Display Settings" 2 times for 8 bit interface
+    writeByte(0x30);
+    wait(0.0041);  // this command takes 4.1ms, so wait for it
+    writeByte(0x30);
+
+
+    writeByte(0x38);     // 8-bit mode, 2 lines 8x5 font
+    //wait(0.000040f);    // most instructions take 40us
+    writeByte(0x08);  //display off, cursor off
+    writeByte(0x01);  //clear display
+    writeByte(0x06);  //increment, no shift
+//from here on busy can be tested
+
+    writeCommand(0x0F); //display on, cursor on, blink on
+//    cls();
+::printf("lcd created\n");
+}
+
+int TextLCD23017::printf (char * format, ...) {
+    char buffer[40];
+    va_list args;
+    va_start (args, format);
+    int rv=vsprintf (buffer,format, args);
+//    ::printf("printing:'%s'\n", buffer);
+    writeString (buffer);
+    va_end (args);
+    return rv;
+}
+
+//busy not used, i2c is slow enough
+bool TextLCD23017::busy() {//assume PORT_A is input, PORT_A intr disabled, takes minimum 10bytes ~1ms
+//setup and hold of RS-E are not met!
+    _intf.write(PORT_B, RW|E|KEYS); //read R=1, E=1, disable keys
+    _ar = _intf.read(PORT_A);
+    _intf.write(PORT_B, KEYS); //read R=0, E=0
+    return (_ar & 0x80) ;//return with keys disabled
+}
+
+void TextLCD23017::character(int column, int row, int c) {
+    int a = address(column, row);
+    writeCommand(a);
+    writeData(c);
+}
+
+void TextLCD23017::cls() {
+    writeCommand(0x01); // cls, and set cursor to 0
+    wait(0.00164f);     // This command takes 1.64 ms
+    locate(0, 0);
+}
+
+void TextLCD23017::locate(int column, int row) {
+    _column = column;
+    _row = row;
+}
+
+int TextLCD23017::_putc(int value) {
+    if (value == '\n') {
+        _column = 0;
+        _row++;
+        if (_row >= rows()) {
+            _row = 0;
+        }
+    } else {
+        character(_column, _row, value);
+        _column++;
+        if (_column >= columns()) {
+            _column = 0;
+            _row++;
+            if (_row >= rows()) {
+                _row = 0;
+            }
+        }
+    }
+    return value;
+}
+
+int TextLCD23017::_getc() {
+    return -1;
+}
+
+void TextLCD23017::writeByte(int value) {//5 transactions of 3 bytes ~150bits ~1.5ms
+    _intf.direction(PORT_A, PORT_DIR_OUT); //set to output
+    _intf.write(PORT_A, value);//put value on data bus
+    _intf.write(PORT_B, E|KEYS);//write instruction register, E=1
+    _intf.write(PORT_B, KEYS);//E returns to 0
+    _intf.direction(PORT_A, PORT_DIR_IN); //set to input
+}
+
+void TextLCD23017::writeCommand(int command) {//3 transactions 3 bytes ~0.9ms + busy + writeByte = 3.4ms
+//    while (busy()) /*wait*/ ;
+    writeByte(command);
+    _intf.write(PORT_B, 0);//enable keys
+}
+
+void TextLCD23017::writeData(int data) {
+//    while (busy()) /*wait*/ ;
+    _intf.direction(PORT_A, PORT_DIR_OUT); //set to output
+    _intf.write(PORT_A, data);//put data on data bus
+    _intf.write(PORT_B, RS|KEYS);//setup RS
+    _intf.write(PORT_B, E|RS|KEYS);//write data register, E=1, RS=1
+    _intf.write(PORT_B, RS|KEYS);//E returns to 0
+    _intf.direction(PORT_A, PORT_DIR_IN); //set to input
+    _intf.write(PORT_B, 0);//enable keys
+}
+
+void TextLCD23017::writeString(char *s) {//convert a string into LCD commands
+    char buffer[240];//size must be > 4*strlen(s) + 7
+    char *p = strtok(s,"\n");
+    _intf.direction(PORT_A, PORT_DIR_OUT); //set to output
+    //assume RS=0, E=0, R/W = 0
+    while (p) {
+//    ::printf("section:'%s'\n", p);
+        int i = 0;
+        if (p != s) {//don't do this the first time (not a newline)
+            _column = 0;
+            if (++_row >= rows()) _row = 0;
+        }
+        buffer[i++] = E|KEYS;//B, clock RS and RW
+        buffer[i++] = address(_column, _row);//A, this data is used
+        buffer[i++] = KEYS;//B, clock data
+        buffer[i++] = address(_column, _row);//A, dummy
+        buffer[i++] = RS|KEYS;//B, setup RS
+        for (int j = 0; j < strlen(p) && i<sizeof(buffer)-4; j++) {
+            buffer[i++] = p[j];//A, dummy
+            buffer[i++] = E|RS|KEYS;//B, clock RS and RW
+            buffer[i++] = p[j];//A, this data is used
+            buffer[i++] = RS|KEYS;//B, clock data
+            if (++_column >= columns()) {
+                _column = 0;
+                if (++_row >= rows())  _row = 0;
+            }
+        }//RS=1, E=0, RW=0
+        p = strtok(0, "\n");
+        buffer[i++] = 0;//A, dummy
+        buffer[i++] = (p!=0) ? KEYS : 0; //B, release RS
+        _intf.write(PORT_B, buffer, i, p != 0);//start with B, send all substrings as a single transaction
+    }
+    _intf.direction(PORT_A, PORT_DIR_IN); //set to input
+}
+
+int TextLCD23017::address(int column, int row) {
+    switch (_type) {
+        case LCD20x4:
+            switch (row) {
+                case 0:
+                    return 0x80 + column;
+                case 1:
+                    return 0xc0 + column;
+                case 2:
+                    return 0x94 + column;
+                case 3:
+                    return 0xd4 + column;
+            }
+        case LCD16x2B:
+            return 0x80 + (row * 40) + column;
+        case LCD16x2:
+        case LCD20x2:
+        default:
+            return 0x80 + (row * 0x40) + column;
+    }
+}
+
+int TextLCD23017::columns() {
+    switch (_type) {
+        case LCD20x4:
+        case LCD20x2:
+            return 20;
+        case LCD16x2:
+        case LCD16x2B:
+        default:
+            return 16;
+    }
+}
+
+int TextLCD23017::rows() {
+    switch (_type) {
+        case LCD20x4:
+            return 4;
+        case LCD16x2:
+        case LCD16x2B:
+        case LCD20x2:
+        default:
+            return 2;
+    }
+}