1 /* 2 * Author: Mihai Tudor Panu <mihai.tudor.panu (at) intel.com> 3 * Copyright (c) 2015 Intel Corporation. 4 * 5 * Permission is hereby granted, free of charge, to any person obtaining 6 * a copy of this software and associated documentation files (the 7 * "Software"), to deal in the Software without restriction, including 8 * without limitation the rights to use, copy, modify, merge, publish, 9 * distribute, sublicense, and/or sell copies of the Software, and to 10 * permit persons to whom the Software is furnished to do so, subject to 11 * the following conditions: 12 * 13 * The above copyright notice and this permission notice shall be 14 * included in all copies or substantial portions of the Software. 15 * 16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 17 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 18 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 19 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE 20 * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION 21 * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION 22 * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 23 */ 24 25 #include <string> 26 #include <stdexcept> 27 #include "tm1637.h" 28 #include <stdarg.h> 29 30 const uint8_t m_brkt[2] = {0x39, 0x0f}; 31 const uint8_t m_nums[10] = {0x3f, 0x06, 0x5b, 0x4f, 0x66, 32 0x6d, 0x7d, 0x07, 0x7f, 0x6f}; 33 const uint8_t m_char[26] = {0x77, 0x7c, 0x39, 0x5e, 0x79, 34 0x71, 0x6f, 0x76, 0x30, 0x1e, 35 0x00, 0x38, 0x00, 0x00, 0x5c, 36 0x73, 0x67, 0x50, 0x5b, 0x78, 37 0x3e, 0x1c, 0x00, 0x00, 0x6e, 38 0x5b}; 39 40 using namespace std; 41 using namespace upm; 42 43 upm::TM1637::TM1637(int clk_pin, int dio_pin, int bright, M_FAST_GPIO mmio) { 44 45 if((m_clk = mraa_gpio_init(clk_pin)) == NULL){ 46 throw std::invalid_argument(std::string(__FUNCTION__) + 47 ": mraa_gpio_init(clk) failed, invalid pin?"); 48 return; 49 } 50 51 if((m_dio = mraa_gpio_init(dio_pin)) == NULL){ 52 throw std::invalid_argument(std::string(__FUNCTION__) + 53 ": mraa_gpio_init(dio) failed, invalid pin?"); 54 return; 55 } 56 57 mraa_gpio_dir(m_clk, MRAA_GPIO_OUT); 58 mraa_gpio_dir(m_dio, MRAA_GPIO_OUT); 59 60 // Let the resistors pull the lines high 61 mraa_gpio_mode(m_clk, MRAA_GPIO_PULLUP); 62 mraa_gpio_mode(m_dio, MRAA_GPIO_PULLUP); 63 64 if(mmio){ 65 if(mraa_gpio_use_mmaped(m_clk, 1) != MRAA_SUCCESS || 66 mraa_gpio_use_mmaped(m_dio, 1) != MRAA_SUCCESS){ 67 throw std::runtime_error(std::string(__FUNCTION__) + 68 ": mraa_gpio_use_mmaped() failed"); 69 return; 70 } 71 } 72 73 mraa_gpio_write(m_clk, 0); 74 mraa_gpio_write(m_dio, 0); 75 76 for (int i = 0; i < M_DISPLAY_DIGITS; i++) { 77 m_digits[i] = 0x00; 78 } 79 setBrightness(bright); 80 } 81 upm::TM1637::~TM1637() { 82 for (int i = 0; i < M_DISPLAY_DIGITS; i++) { 83 m_digits[i] = 0x00; 84 } 85 update(); 86 87 mraa_gpio_close(m_clk); 88 mraa_gpio_close(m_dio); 89 } 90 mraa_result_t upm::TM1637::write(uint8_t *digits) { 91 for (int i = 0; i < M_DISPLAY_DIGITS; i++) { 92 m_digits[i] = digits[i]; 93 } 94 update(); 95 return MRAA_SUCCESS; 96 } 97 mraa_result_t upm::TM1637::write(int d, ...) { 98 va_list args; 99 va_start(args, d); 100 m_digits[0] = (uint8_t)d; 101 102 for (int i = 1; i < M_DISPLAY_DIGITS; i++) { 103 m_digits[i] = (uint8_t)va_arg(args, int); 104 d++; 105 } 106 va_end(args); 107 update(); 108 return MRAA_SUCCESS; 109 } 110 mraa_result_t upm::TM1637::writeAt(int index, char symbol) { 111 if(index < 0 || index >= M_DISPLAY_DIGITS){ 112 cerr << "TM1637: invalid index in " << __FUNCTION__ << endl; 113 return MRAA_ERROR_INVALID_PARAMETER; 114 } 115 m_digits[index] = encode(symbol); 116 update(); 117 return MRAA_SUCCESS; 118 } 119 mraa_result_t upm::TM1637::write(std::string digits) { 120 int len = digits.length(); 121 if( len > M_DISPLAY_DIGITS){ 122 len = M_DISPLAY_DIGITS; 123 } 124 for (int i = 0; i < len; i++) { 125 m_digits[i] = encode(digits[i]); 126 } 127 update(); 128 return MRAA_SUCCESS; 129 } 130 void upm::TM1637::setColon(bool value) { 131 if(value){ 132 m_digits[1] |= 0x80; 133 } 134 else{ 135 m_digits[1] &= 0x7f; 136 } 137 update(); 138 } 139 void upm::TM1637::setBrightness(int value) { 140 m_brightness = value & 0x07; 141 update(); 142 } 143 void upm::TM1637::i2c_start() { 144 mraa_gpio_write(m_clk, 1); 145 mraa_gpio_write(m_dio, 1); 146 mraa_gpio_write(m_dio, 0); 147 } 148 void upm::TM1637::i2c_stop() { 149 mraa_gpio_write(m_clk, 0); 150 mraa_gpio_write(m_dio, 0); 151 mraa_gpio_write(m_clk, 1); 152 mraa_gpio_write(m_dio, 1); 153 } 154 void upm::TM1637::i2c_writeByte(uint8_t value) { 155 for(uint8_t i = 0; i < 8; i++) 156 { 157 mraa_gpio_write(m_clk, 0); 158 if(value & 0x01) 159 mraa_gpio_write(m_dio, 1); 160 else 161 mraa_gpio_write(m_dio, 0); 162 value >>= 1; 163 mraa_gpio_write(m_clk, 1); 164 } 165 166 // Ack clock without skew, TM1637 is fast enough 167 mraa_gpio_write(m_clk, 0); 168 mraa_gpio_write(m_clk, 1); 169 mraa_gpio_write(m_clk, 0); 170 } 171 void upm::TM1637::update() { 172 i2c_start(); 173 i2c_writeByte(TM1637_ADDR); 174 i2c_stop(); 175 176 i2c_start(); 177 i2c_writeByte(TM1637_REG); 178 for (int i = 0; i < M_DISPLAY_DIGITS; i++) { 179 i2c_writeByte(m_digits[i]); 180 } 181 i2c_stop(); 182 183 i2c_start(); 184 i2c_writeByte(TM1637_CMD | m_brightness); 185 i2c_stop(); 186 } 187 uint8_t upm::TM1637::encode(char c) { 188 if(c >= '0' && c <= '9') 189 return m_nums[(int)c - 48]; 190 if(c >= 'a' && c <= 'z') 191 return m_char[(int)c - 97]; 192 if(c >= 'A' && c <= 'Z') 193 return m_char[(int)c - 65]; 194 if(c == '[') 195 return m_brkt[0]; 196 if(c == ']') 197 return m_brkt[1]; 198 if(c == '(' || c == ')') 199 return m_brkt[(int)c - 40]; 200 if(c == '-') 201 return 0x40; 202 if(c == '_') 203 return 0x08; 204 if(c == '}') 205 return 0x70; 206 if(c == '{') 207 return 0x46; 208 return 0x00; 209 } 210