Пробуем организовать связь с TM1637 без библиотеки.
- Войдите на сайт для отправки комментариев
Поставил я себе невыполнимую задачу написать первую свою программку для ардуины. Взор упал на датчик DHT11. За три дня удалось наладить общение с датчиком без библиотеки. Обрадовался я и поставил себе еще более сложную задачу для новичка: наладить связь 4-разрядным дисплеем, где в качестве драйвера микросхема TM1637. Покурил DATASHEET, наваял за недельку в свободное время код.
byte command1 = B01000000; //Команда записи данных в регистр дисплея byte command2 = B11000000; //Команда "начальный адрес дисплея" byte symbol_1 = 0xF0;//Символ_1 byte symbol_2 = 0x20;//Символ_2 byte symbol_3 = 0xF0;//Символ_3 byte symbol_4 = 0x80;//Символ_4 byte symbol_5 = 0xF0;//Символ_5 byte symbol_6 = 0xD0;//Символ_6 byte command3 = B10001101; //Яркость и ON/OFF дисплея byte command4 = B01000010;//Команда чтения клавиатуры. unsigned int bit_n; unsigned int WAIT_HL = 0; unsigned int pack = 0x01; unsigned int send_byte; //int SCL_LOW = 0; //Переменная, в которой хранится номер цикла при ожидании LOW на SCL. //int SCL_HIHG = 0; //Переменная, в которой хранится номер цикла во время ожидания HIGH SCL. unsigned int SCL_TM1637 = B00001000; //Задаем 3 пин, как выход SCL unsigned int SDA_TM1637 = B00000100; //Задаем 2 пин, как шину SDA volatile boolean SCL_STAT = true;//Переменная, в которой хранится текущее состояние выхода SCL volatile boolean ACK_tm1637 = false; //Переменная в которой возвращается проверка сигнала ACK (подтверждение приема байта) void setup() { TIMSK2 &= ~(1<<TOIE2);// Запрещаем прерывания, сбросив бит TOIE2 в регистре TIMSK2. TCCR2A &= ~((1<<COM2A1) | (1<<COM2A0) | (1<<COM2B1) | (1<<COM2B0) | (1<<WGM21) | (1<<WGM20));//Сбрасываем биты 7 - 4 в регистре TCCR2A, так как переключение выходов OC2A и OC2B нас пока не интересует, а также отключаем режим ШИМ (WGM). TIMSK2 |= (1<<TOIE2);// Разрешаем прерывания, установив бит TOIE2 в регистре TIMSK2. Прерывание будет переворачивать сигнал на SCL DDRD |= B00001100; //Инициализируем третий и второй пины порта D как выходы. PORTD |= B00001100; //Выводим HIGH на третий и второй пины порта D. Serial.begin(9600); } void generator_SCL(){ PORTD = PORTD ^ SCL_TM1637; //Инвертируем третий пин порта D. Тем самым формируем сигнал SCL. //Serial.println(PORTD); SCL_STAT = !SCL_STAT; //Записываем состояние SCL в переменную SCL_STAT } void start_transfer(){ DDRD |= B00001100; //Инициализируем третий и второй пины порта D как выходы. PORTD |= B00001100; //Выводим HIGH на SCL и SDA SCL_STAT = true; delay(1); PORTD &= B11111011; // Посылаем на SDA LOW, тем самым отправляем приемнику сигнал о старте передачи данных (SCL = HIGH). TCCR2B = 0x02; //Подключаем 2 Мгц к Таймеру2. Таймер начинает считать до 255, вызывать подпрограмму прерывания (generator_SCL()) и сбрасываться в 0. } void stop_start(){ //Используется после инициализации и перед третьей командой WAIT_HL = 0; while(WAIT_HL < 2000){ //Ждем LOW на SCL (см. DATASHEET TM1637 if(!(PIND & B00001000)){ //Если пришел LOW, //Serial.println("GOOD"); WAIT_HL = 2000; //идем дальше } WAIT_HL += 1; } WAIT_HL = 0; PORTD |= SDA_TM1637; //выводим 1 на пин SDA, while(WAIT_HL < 50){ WAIT_HL += 1; } WAIT_HL = 0; PORTD &= B11111011; //выводим 0 на пин SDA, } void end_transfer(){ TCCR2B = 0x00; //Останавливаем Таймер2. PORTD |= B00001000; //Выводим на SCL HIGH SCL_STAT = true; // Не забываем присвоить переменной статуса SCL значение true DDRD |= B00000100; //Инициализируем второй пин порта D (SDA) как выход. PORTD &= B11111011; //Выводим на SDA LOW. delay(1); TCNT2 = 0; //Обнуляем таймер. PORTD |= B00000100; //Выводим на SDA HIGH. } void wait_ack(){ ACK_tm1637 = 0; WAIT_HL = 0; while(WAIT_HL < 2000){ //Ждем LOW на SCL (см. DATASHEET TM1637 if(!SCL_STAT){ //Если пришел LOW, //Serial.println("GOOD"); WAIT_HL = 2000; //идем дальше } WAIT_HL += 1; } DDRD &= B11111011; //Инициализируем пин SDA как вход PORTD |= B00000100; //Подключаем подтягивающий резистор к SDA WAIT_HL = 0; while(WAIT_HL < 2000){ //Начинаем слушать шину SDA не более 2000 циклов //Serial.println(PORTD); if(!(PIND & B00000100)){//Если SDA = LOW, Serial.println("OK"); Serial.println(pack); WAIT_HL = 2000; ACK_tm1637 = 1; //присваиваем переменной ACK_tm1637 значение 1, как подтверждение получения ACK } WAIT_HL += 1; } if(ACK_tm1637 == 0){ Serial.println(pack); Serial.println("NO ACK!"); } WAIT_HL = 0; while(WAIT_HL < 2000){ //Ждем положительного фронта на SCL (см. DATASHEET TM1637 if(PIND & B00001000){ //Если пришел положительный фронт, //Serial.println("GOOD"); WAIT_HL = 2000; //уходим в loop } WAIT_HL += 1; } WAIT_HL = 0; DDRD |= B00000100; //Инициализируем второй пин (SDA) порта D как выход. } void byte_send(){ send_byte = pack; //DDRD |= B00000100; //Инициализируем второй пин (SDA) порта D как выход. bit_n = 0; while(bit_n <= 7){ //отправляем байт на SDA //Serial.println(pack); //Serial.println(bit_n); if(!SCL_STAT){ //Если пин SCL переключился на LOW //Serial.println(pack); if(send_byte & B00000001){ //Если младший бит равен 1, //Serial.println("1"); PORTD |= SDA_TM1637; //выводим 1 на пин SDA, send_byte = send_byte >> 1; //сдвигаем байт на 1 бит вправо и переходим к отправке следующего бита. bit_n += 1; //i += 1; while(!SCL_STAT){} //Ждем HIGH на SCL } else{ //Если младший бит равен 0 //Serial.println("0"); PORTD &= ~SDA_TM1637; //выводим на SDA 0, send_byte = send_byte >> 1; //сдвигаем байт вправо на 1 бит и переходим к отправке следующего бита bit_n += 1; // z +=1; while(!SCL_STAT){} //Ждем HIGH на SCL } } } //Serial.println(pack); //wait_ack(); //После передачи байта переходим в подпрограмму ожидания сигнала подтверждения приема } void loop() { //delay(1000); //Serial.println(pack); start_transfer(); pack = command1; byte_send(); wait_ack(); stop_start(); pack = command2; byte_send(); wait_ack(); pack = byte(random()); byte_send(); wait_ack(); pack = byte(random()); byte_send(); wait_ack(); pack = byte(random()); byte_send(); wait_ack(); pack = byte(random()); byte_send(); wait_ack(); //pack = symbol_5; //byte_send(); //wait_ack(); //pack = symbol_6; //byte_send(); //wait_ack(); stop_start(); pack = command3; byte_send(); wait_ack(); pack = command4; byte_send(); wait_ack(); end_transfer(); delay(5000); } ISR(TIMER2_OVF_vect) { generator_SCL(); }
А работать он не хочет. Хотя сигнал ACK от TM1637 приходит. Похоже, я не совсем разобрался в структуре команд данной микросхемы. Замечена следующая странность. Если залить в Ардуино простейший скетч с библиотекой для работы с TM1637, подождать вывода на экран каких-нибудь цифр, а потом залить мой код, то с каждой отправкой пакетов на дисплее начинают менять кракозябры. Видео https://www.youtube.com/watch?v=XTGoT2qXfXs
Причем, такое происходит, даже если отправлять постоянные, а не рандомные значения. Тоесть, в память TM1637 что-то записывается , но что и по какой логике - не понятно. Если сделать рестарт Ардуины, то на дисплее не светится ни один сегмент.
Не совсем понятен следующий момент из даташита. Что значит инициализация? Правильно ли я понял, что это команда 0100 0000? В блок-схеме получается три команды: инициализация, команда записи в память, начальный адрес. А на графике две команды.
Еще заметил особенность. Если использовать в качестве символов использовать константы, то информация на дисплее (в случае если его "завести" предварительно стандартной библиотекой) сдвигается влево при каждом сеансе связи с TM1637. (кракозябры бегут влево). Видео https://www.youtube.com/watch?v=xdhpGZvaWU0 И это происходит, даже если в 126 строке записать while bit_n < 6 Тоесть даже если отправлять не 8 бит, а 5. Получается, микросхема просто сдвигает данные в своей памяти на 1 адрес при каком-то событии на SCL и SDA, а программа моя не работает.
///завести" предварительно стандартной библиотекой
Она просто правильно его инициализирует. Ищите ошибку в инициализации. Но учитывая что она идет первой, то может ошибатся и в низкоуровневом обмене, проверяйте порядок отправки бит например.
Остальное - потом.
У мег есть шикарный аппаратный драйвер шины I2C, пользуйтесь им и не мучайтесь. Работает вплоть до 880кГц, позволяет меге быть как мастером, так и слейвом, и даже отправлять пакеты типа "всем" и принимать таковые .. зачем это?!?
Завязыватся на драйвер конкретного процессора не имеет смысла, сегодня один процессор завтра другой и снова изучать его аппаратную часть заново прийдется. При том что в некоторых драйвера вообще нет. А научитесь работать с i2c низкоуровнево - это умение на всю жизнь полезное, у любого процессора на любых ногах заработает Ваш код после правки буквально 5 строк.
Просто спортивный интерес. Хочется разобраться, как работает TM1637. Ну и I²C. Хоть в этом случае он и неполноценный.
Ой, а программа то работает, оказывается. Если ввести вот такие символы
Krendelyok
Постараюсь обойтись без оценок, но вы _все_ делаете "не так". Техдок на TM163x в переводе на английский действительно выглядит невразумительно. Но способы работы с "подтянутой" линией никто не отменял. И чтобы два раза не вставать - приведенный вами код никуда не годится, причем это не критика кода, а критика подхода. Коротко она (критика) выглядит так:
- если вы беретесь генерировать тактовые сигналы по SCL (в вашем коде через прерывание таймера), то обеспокойтесь правильным (по времени) выставлением уровней по линии SDA (данные)
- на "подтянутых" линиях (которые соединены с питанием через подтягивающие резисторы) используют две операции: освободить линию (перейти в высокоимпеденсное состояние, т.е. отключиться от линии, и тогда резисторы подтянут ее в состояние логической единицы), и прижать линию к земле, установив на ней логический ноль. Что-либо писать в подтянутую линию (устанавливая бит порта в "1") смысла не имеет
Ваш код не делает этого.
Отдельно вопросы с работой самой TM163x. Видимо мы читали одинаковую документацию, но поняли логику по-разному. Там все работает согласно описанию (чуднОму в переводе, но все-таки).
Режимы "бегущей строки", шаманство с "номером" команды - все из области ударов в бубен, заменяющих работу по техдоку. Если вам еще интересно разобраться, могу предоставить примеры кода.
Отдельно по рекомендациям использовать аппаратные (да и программные) драйверы I2C. Посадить TM163x на общую шину I2C нельзя. Работать с TM163x строго по спецификациям I2C - излишне.
Krendelyok
могу предоставить примеры кода.
Спасибо за ответ. Нашел в интернете вот такой код http://xn--90azbaece.xn--p1ai/?p=97 В принципе, всё понятно, вроде бы. Понятно, как просто можно запрограммировать I²C на любом пине. Сразу расхотелось работать с таймером. Бегло пробежался по коду и комментариям. Пока не понял, зачем делается очистка дисплея. Без нее, похоже, не работает ничего. Ткните носом, где в даташите об этом говорится? Во всяком случе выдернул из этого кода TM1637_writeByte, TM1637_stop(), TM1637_start(). Но этих функций недостаточно для того, чтобы дисплей работал. Тоесть работа с TM1637 идет в такой последовательности:
Но чего-то не хватает для того, чтобы дисплей зажегся.
Уточните, пожалуйста, к какой конкретно строчке кода это относится? А то голова уже не соображает.
Krendelyok
Ok. Давайте так - никто вас тыкать ни во что не будет, потому что вы всё делаете в правильном направлении (и очень быстро), но на пути невразумительный даташит и путаница (не ваша) с I2C. Сейчас мы просто разложим по косточкам (на предмет "понять как") работать с китайской микросхемой TM163x. Их целое (и интересное) семейство, управляемое на один лад.
Мне понадобится некоторое время (часов 12), чтобы аккуратно всё сформулировать и привести примеры.
У мег есть шикарный аппаратный драйвер шины I2C, пользуйтесь им и не мучайтесь. Работает вплоть до 880кГц, позволяет меге быть как мастером, так и слейвом, и даже отправлять пакеты типа "всем" и принимать таковые .. зачем это?!?
Krendelyok
Будет много букв, по-другому не получилось. Это первая часть.
1. Терминология
На линиях "с подтяжкой к питанию" (pullup) используют операции:
- освободить линию (перейти в высокоимпедансное состояние, т.е. отключиться от линии), что приводит к установлению на линии логической единицы (работают подтягивающие к питанию резисторы). Это означает перевод пина МК в состояние (input, low), т.е. DDR(PIN) -> 0, PORT(PIN) -> 0
- прижать линию к земле (установить на линии логический ноль). Это означает перевод пина МК в состояние (output, low), т.е DDR(PIN) -> 1, PORT(PIN) -> 0
В своих примерах буду использовать следующие "ясно читаемые" макросы, заимствованные из апноута avr035 "Efficient C Coding for AVR" (просто нравятся):
#define SETBIT(ADDRESS,BIT) (ADDRESS |= (1<<BIT))
Использование этих макросов предполагает определение конкретной ножки МК, например
#define MY_LED_PIN PORTD, PORTD5
и, собственно, манипулирование
C_SETBIT(MY_LED_PIN); // установить 5-й пин порта D в логическую "1"
C_CLEARBIT(MY_LED_PIN); // установить 5-й пин порта D в логический "0"
2. Операции
Для работы с TM163x нужно задать две ноги МК - пин SDA, по которому передаются (принимаются, об этом позже) данные, и пин SCL, который будет тактирующим ("часы") для данных. Это обычная синхронная двухпроводная шина передачи данных. Таким же образом, напрмер, мы передаем данные в сдвиговые регистры. Способ передачи данных - сначала выставляем на ноге SDA данные (логический ноль или логическую единицу), затем даем синхроимпульс по ноге SCL (поднимаем ее в логическую единицу, выдерживаем некоторое время, опускаем в логический ноль; считывание принимающим устройством происходит в момент перевода SCL в логическую единицу).
Для манипулирования входами TM163x определим удобные (удобочитаемые) макросы
а) определим ножки МК, которые будут подключаться к входам DIO и CLK (по техдоку) микросхемы TM163x
например ножка PORTD4 порта D будет источником синхроимпульсов ("часами"), а по ножке PORTD5 порта D будут передаваться данные
б) и вот теперь определим то, чего мы хотим от этих ножек МК
3. Использование
Зачем все эти навороты из макросов?
Ниже пойдут примеры работы с микросхемой TM163x, которая понимает состояние своих входов. Этими входами нужно уметь управлять. Важно понимать, как формировать логические "0" и "1".
Установить логическую "1" на подтянутой (к питанию, pullup) линии означает ничего не делать, просто отключиться.
Установить логический "0" на подтянутой (к питанию, pullup) линии означает прижать ее к земле.
Это повторение пункта 1. Макросы из п.2 показывают как именно это делается. Можно написать свои макросы, имеющие другой вид. Нужно держать в памяти следующее - тот, с кем вы работаете (у нас это микросхема TM163x), реагирует на состояние линии - HIGH или LOW. Состояние (подтянутой к питанию, pullup) линии МК определяется манипуляцией пином - регистами DDR и PORT этого пина. Если мы хотим получить "0" (LOW) на линии, мы обеспечиваем DDR (пина) -> 1 (output для пина) и PORT (пина) -> 0 (прижимаем линию к земле). Если мы хотим получить "1" (HIGH) на линии, мы обеспечиваем DDR (пина) -> 0 (input для пина).
В примерах кода буду писать CLK_LOW или, например, DIO_HIGH.
Для работы с TM163x логично написать базовый код, который обеспечивает передачу-прием байта, и логику поверх него.
Базовый код может быть написан (он так и выглядит в моей реализации) таким образом.
Макросы DIO_LOW, CLK_HIGH и CLK_LOW рассмотрены выше, они задают состояние соответствующих линий.
Параметр _CLOCK_DELAY (микросекунды) определяет задержку, чтобы TM1637 работал на своей частоте не выше ~450 kHz. Для быстрых МК (16 MHz) _CLOCK_DELAY = 1 us так оно и выходит (2 х 1us ~ 500 kHz)
Используя три приведенных функции, можно построить работу с TM1637.
Для полноты картины следует привести базовую функцию чтения кнопок. Многие чипы из семейста TM163x могут читать кнопки, замыкающиеся на общий провод.
Параметр _KEY_DELAY (микросекунды) определяет задержку сканирования кнопок, обычно ~0.5ms x число_разрядов, у меня работает при 5us для 16MHz МК (это минимум).
Техдок на TM163x можно резюмировать следующим образом. Комментарии на буржуазном языке отражают не мою англоманию, а сложности с кириллицей в моем IDE.
Из приведенного выше следует, что есть 3 (функциональных) группы команд.
Каждая команда начинается с start(), далее write() (команда и данные), завершается stop().
Инициализация TM1637 (мне непонятно что там со значениями по умолчанию) выглядит так:
Здесь _TM1637_DIGS_ATTACHED константа, определяющая кол-во разрядов (цифр) 7-сегментного индикатора.
И как пример простой вывод положительного числа на индикатор, где dp - позиция десятичной точки, отсчитывая справа.
где digitToSegment2[] таблица перекодировки цифры в представление сегментов индикатора вроде
Да, это не совсем I2C и аппаратный драйвер не подойдет, хотя мне сначала показалось что можно в качестве адреса подавать первый байт команды .. ан нет. режимы R/W в наборе команд не согласуются с младшим битом команды..
1. Терминология
На линиях "с подтяжкой к питанию" (pullup) используют операции:
- освободить линию (перейти в высокоимпедансное состояние, т.е. отключиться от линии), что приводит к установлению на линии логической единицы (работают подтягивающие к питанию резисторы). Это означает перевод пина МК в состояние (input, low), т.е. DDR(PIN) -> 0, PORT(PIN) -> 0
Вот этот момент мне не понятен. По-моему, тут нестыковка с даташитом на атмегу328. На 97 странице есть такая таблица
http://cs636829.vk.me/v636829594/2f006/Ae1kfNW93Ls.jpg Кстати, не совсем понятна третья строчка в этой таблице. Получается, можно сконфигурировать пин как Вход, записать туда единицу, но, записав в PUD = 1, отключить встроенный подтягивающий резистор. Правильно я понимаю, что это состояние не будет отличаться от первой строчки?
Из нее я и сделал вывод. Чтобы начать "слушать" шину SDA, нужно инициализировать пин как вход и Записать в него HIGH. В этом случае мы подключаем встроенный резистор подтяжки. А если запишем LOW, у нас пин перейдет в высокоимпедансное состояние. Тогда нам нужен будет внешний резистор подтяжки. Правильно? Поэтому я и использовал в коде команду записи единицы на вход. Дальше пока не читал, буду разбираться. Наверное, вечером)
Я только вчера узнал, что такое макросы))) Прикольно.
Как выяснилось, для простого вывода цифр на экран достаточно удалить всё лишнее, и остается вот такой простенький код:
Очистка дисплея не нужна (для чего она вообще нужна?). Можно еще таблицу-генератор символов оставить из оригинального кода. Вроде как, кто-то жаловался, что стандартная библиотека tm1637.h не везде работает. Интересно, такой код будет работать, скажем, на тиньке 13? ))
Уф. Осилил много букв. Открыли мне глаза на грамотное программирование)) Это ж этак можно программу своими словами написать, а после расшифровать их в макросах. А главное, читать то как удобно. В принципе, все понял, кроме предыдущего моего вопроса и вот этих строк:
И еще, может, кто попутно даст ссылку, в чем осбенность записи вида void name(void)? Или
unsigned
char
void
tm1637_scan_key(
void
). А то я немного поизучал Java, а потом подсел на ардуино. А тут всё в куче. Не понятно.
unsigned
char
void
tm1637_scan_key(
void
) - это ошибка (описка). Разумеется правильно будет
unsigned
char
tm1637_scan_key(
void
)
Вот этот момент мне не понятен. По-моему, тут нестыковка с даташитом на атмегу328. На 97 странице есть такая таблица
http://cs636829.vk.me/v636829594/2f006/Ae1kfNW93Ls.jpg Кстати, не совсем понятна третья строчка в этой таблице. Получается, можно сконфигурировать пин как Вход, записать туда единицу, но, записав в PUD = 1, отключить встроенный подтягивающий резистор. Правильно я понимаю, что это состояние не будет отличаться от первой строчки?
Из нее я и сделал вывод. Чтобы начать "слушать" шину SDA, нужно инициализировать пин как вход и Записать в него HIGH. В этом случае мы подключаем встроенный резистор подтяжки. А если запишем LOW, у нас пин перейдет в высокоимпедансное состояние. Тогда нам нужен будет внешний резистор подтяжки. Правильно? Поэтому я и использовал в коде команду записи единицы на вход. Дальше пока не читал, буду разбираться. Наверное, вечером)
Включив внутреннюю подтяжку (DDRx->0, PORTx->1), вы определяете "1" на линии. Если "с другой стороны" прижмут линию к земле, от вас потечет ток, чего "другая сторона" может не ожидать. (Pxn will source current if ext. pulled low. из даташита).
Линии данных и синхроимпульсов для TM163x подтягиваются внешними резисторами, их нормальное состояние "1". Именно они (резисторы) вытягивают линии к "1", когда вы со своей стороны перестаете их прижимать к земле ("0"). Состояние линии данных после сгенерированного восходящего фронта синхроимпульса считывается через соответствующий пин регистра PINx.
Различие между I2C и двухпроводным интерфейсом TM163x не только в отсутствие адресации. TM163x не поддерживает master hold (clock stretching), поэтому зачем китайские товарищи встроили ACK - непонятно. Разве чтобы не превышали скорость обмена. В этом смысле телодвижения по поимке ACK в конце передачи байта выглядят излишними, достаточно дополнительного синхроимпульса. (Однако при считывании байта командой scan_key выдать ACK прийдется).
Включив внутреннюю подтяжку (DDRx->0, PORTx->1), вы определяете "1" на линии. Если "с другой стороны" прижмут линию к земле, от вас потечет ток, чего "другая сторона" может не ожидать. (Pxn will source current if ext. pulled low. из даташита).
Так ток потечет точно так же и через внешние резисторы, если линия подтянута к ним. Тут суть не меняется. Просто меняется расположение резисторов. Другое дело, если на линии много устройств, и не у всех есть встроенные резисторы. Тогда логично использовать внешние резисторы подтяжки. А в случае, когда один мастер - не вижу особого смысла во внешних резисторах.
У вас в коде, кстати, инициализация выполняет очистку дисплея. Какой-либо дополнительной функции в ней нет. Как видно из вырезанного мной кода, все работает без очистки дисплея. В блоке инициализации ничего не осталось, кроме установки пинов в режим output. Что не работало в моей программе , пока не разбирался)) Вроде как, делал я все правильно. Наверное, какой то рассинхрон в использовании таймера2)
Различие между I2C и двухпроводным интерфейсом TM163x не только в отсутствие адресации. TM163x не поддерживает master hold (clock stretching), поэтому зачем китайские товарищи встроили ACK - непонятно. Разве чтобы не превышали скорость обмена. В этом смысле телодвижения по поимке ACK в конце передачи байта выглядят излишними, достаточно дополнительного синхроимпульса. (Однако при считывании байта командой scan_key выдать ACK прийдется).
Я думал, что ACK здесь также, как и в полноценном I²C, выполняет роль контроля ошибок. Не пришел ACK после отправки байта - отправляем программу обрабатывать ошибку. В вырезанном из библиотеки коде, кстати, мы видим фрагмент кода с while, где ожидается сигнал ACK (строка 55). Но в нем нет аварийного выхода из while. Тоесть, если ACK не придет, то наша сборка просто зависнет. Может быть, это и вызывает у многих проблему?
P.S. Да, вот здесь уже писали о доработке фрагмента с while http://arduino.ru/forum/programmirovanie/modul-4-led-na-draiver-chipe-tm1637#comment-222621
Только вот функция
void
tm1637_write(uint16_t num, uint8_t dp) вместо того чтоб вывести число 1234, выводит 4321. Ну и точку ставит не там где указано.
Нужно в функции вывода положительного числа
выводить число из массива
digs[_TM1637_DIGS_ATTACHED] начиная с
digs[3], тогда на дисплей все правильно выводится и точка ставится там где указано. Кстати, а как вывести отрицательное число на дисплей?
Доброго вечера всем.
Предыстория. Отец попросил сделать ему "часы с белыми цифрами и чтоб яркость регулировалась", единственное что-то доступное с белыми цифрами- это вот выписаный с Али китайский модуль на tm1637. И яркость как раз умеет. Кстати, кто не вкурсе- модули бывают двух типов: "для часов" с центральными точками, и "приборные" с децимальными- многие по ошибке заказывают "часовые", а потом жалуются что децимальные точки не светят. После того как я почитал ДШ- увидел что микросхема еще и клавиши опрашивать умеет!.. а на модуле их нет... а у меня tiny13- ножек мало... а кнопок надо пять штук... посему- готовый модуль был безжалостно раздерган на детальки. После разводки платы встал вопрос об управлении всем хозяйством, бо для удобства разводки моя схема весьма сильно отличалась от даташитовской. Соотвественно, стандартные библиотеки использовать невозможно, кроме того- я вообще использую Code Vision (привык).
Хочу выразить огромную благодарность пользователю dhog1 ! За предоставленные примеры функций, бо я уже весь инфернет перерыл и отчаялся, а тут такой подарок! Тем не менее, мне они показались неудобны, поскольку я больше "железячник" чем "программист", и использование дефайнов и макросов до сих пор вызывает жуткую неприязнь- неудобны мне они, путаюсь! Ладно б сам писал, а тут еще и чужие... Соотвественно, для себе подобных выкладываю "переделаные" в максимально железячный вид примеры функций с максимально подробными комментариями (писал для себя- то есть как для идиота). Функции совершенно точно рабочие- часы уже стоят на столе и показывают что положено. Коды нажатых клавиш полностью соответствуют кодам по ДШ (если кнопки включены по ДШ- между линиями K и SG)- то есть считывание происходит корректно. Господ программистов убедительно прошу не плеваться что не по-феншую. Надеюсь, кому-нибудь это пригодится- при необходимости крайне легко перепилить под свой проект. Для Code Vision AVR