Шина I2C и AVR пример, PCF8583 (RTC) вечный календарь

Шина I2C очень распространена, существуют АЦП, ЦАП, термометры, память с интерфейсом I2C. При проектировании устройств и их починке приходится сталкиваться с этой шиной. Рассмотрим ее поподробнее. Шина I2C была спроектирована компанией Philips в начале 80-х и предназначена для низкоскоростной передачи данных по двум сигнальным проводникам, SDA для данных и SCL для тактирования.

Для подтяжки линий SDA, SCL используются резисторы номиналом 4.7кОм (при большой емкости шины номинал подтягивающих резисторов требуется уменьшить). Для успешного обмена данными на шине обязательно присутствие одного ведущего устройства (master device), которое производит тактирование и одного или более ведомых устройств (slave device). Количество ведомых устройств ограничено емкостью шины 500пФ и адресным пространством. Каждое ведомое устройство на шине должно иметь8-ми битный адрес, по которому оно отзывается.
При начале обмена данными по I2C, ведущему устройству необходимо сгенерировать так званое условие старта (start condition), а при окончании – условие остановки (stop condition). Эти условия выглядят так:
схема start, stop condition
При старте на линию SDA подается низкий уровень, при высоком уровне на SCL и удерживается так в течении некоторого времени. При стопе линия SDA отпускается до высокого уровня, при высоком уровне на SCL. За конкретными таймингами нужно лезть в спецификацию шины I2C. После подачи условия старта, необходимо подать 8-ми битный адрес устройства, с которым мы хотим продолжить «общение». Это делается вот так:
схема передачи байта
Начиная со старшего бита, подаем на линию SDA данные, после записываем их импульсом на линии SCL, и так 8 раз. После чего ведомое устройство, адрес которого был назван, должно откликнуться, для этого предназначен 9-й синхроимпульс на линии SCL, во время которого адресуемое устройство должно удерживать логический ноль на линии данных SDA, тем самым обозначает свое наличие. Это подтверждение в документации называется ACK (Acknowledge) и используется каждый раз после передачи байта. Подавать сигнал ACK должно устройство которому предназначается команда или данные. То есть, если мы записываем данные в ведомого, откликается ведомый, а если считываем данные – то откликаться должен ведущий, после корректного приема байта. Дальнейший обмен данными производиться по той же схема, 8 бит данных, 1 для ACK и т.д. На словах работа с шиной проста. Предлагаю сварганить простые часы на микросхеме вечного календаря PCF8583, с интерфейсом . Выводить время и дату будет на символьный ЖКИ WH1602A с помощью микроконтроллера ATmega8. Рассмотрим подключение шины I2C к микроконтроллеру.

Тактироваться микроконтроллер будет от внутреннего RC генератора частотой 4 МГц. Кварц Х1 – часовой на 32768Гц, конденсатор С1 – 30пФ, С5,С7 – электролиты на 100 и 200мкФ соответственно, С4,С6 – керамика на 0,1мкФ. Подтягивающие резисторы R1,R4 – 4,7кОм. Ограничительные резисторы R2 – 1кОм, R3 – 17 Ом, потенциометр для регулировки контраста ЖКИ VR1 – 10кОм. Три кнопки послужат для задание даты и времени.
А вот как это выглядит на макетке:
фото макетки с PCF8583
В этой заметке я хочу использовать программный I2C, так как морочиться с аппаратным пока нет желания. Не вдаваясь в подробности работы PCF8583, скажу, что если к ней присоединить кварц с конденсатором и подать питание, то с ее регистров можно будет считать время и дату в BCD формате. Т.е. 23 часа 45 минуты 11 секунд будет закодировано вот так: 0х23 часа 0х45 минуты 0х11 секунд. Кажется удобным, но мне чего-то не понравилось.
распиновка PCF8583
PCF8583 кроме стандартного питания и SDA, SCL имеет ножку А0, которая используется для задания одного из битов ее адреса, для возможности работы двух PCF8583 на одной шине. Также имеется ножка INT, которая может сгенерировать низкий уровень при наступлении некоторого события. Так, как на шине PCF8583 всего одна и всякие события использоваться не будет, то смело подключим А0 к земле, а INT не будем задействовать. Адрес PCF8583 состоит из 2-х тетрад, неизменяемой и изменяемой:
адрес PCF8583
Для того, чтобы записать данные в PCF8583 (в нашем случае А0=0), нужно выдавать адрес 0хА0. Для чтения же, нужно выдавать адрес 0хА1. Немного неудобно. Интересующее время/дата храниться в регистрах начиная с адреса 0х02 и заканчивая адресом 0х06:
данные PCF8583
То есть, наша задача периодически считывать эти данные и выводить их на ЖКИ, предусмотрев возможно записи введенных с клавиш, данных в саму PCF8583. Считывать данные с PCF8583 нужно в таком порядке:
чтение PCF8583
Выдаем на шину условие старта, выдаем адрес для записи 0хА0, выдаем адрес регистра, начиная с которого хотим считать данные, в нашем случае это 0х02. Далее снова выдаем условие старта, адрес устройства для чтения 0хА1 и считываем по одному байты из PCF8583. При последнем считанном байте не нужно давать ACK (этого делать я не стал, т.е. все время даю сигнал ACK, на работоспособность это не повлияло).
Также просмотрим диаграмму для записи данных в PCF8583:
запись PCF8583
Здесь все проще. Выдаем адрес 0хА0, после чего выдаем адрес регистра, начиная с которого хотим производить запись. После чего, один за другим, передаем байты.
Для реализации этих диаграмм напишем следующие функции:

  1. #define SDA 0
  2. #define SCL 1
  3.  
  4. #define READ_PCF 0xa1 //Адрес для чтения
  5. #define WRITE_PCF 0xa0 //Адрес для записи
  6.  
  7. ////edition by МIROSLAV SAMES
  8. //#define READ_PCF 0xA3 //
  9. //#define WRITE_PCF 0xa2 //
  10. //to test replace by two strings below
  11.  
  12. unsigned char ack=0; //АСК, если=1 – произошла ошибка
  13. unsigned char data[5]={}; //Данные время/даты
  14. DDRB=0x03; //init PORTB
  15. PORTB=0x0c;
  16. void start_cond (void) //Генерация условия старта
  17. {
  18. PORTB=_BV(SDA)|_BV(SCL);
  19. asm("nop");
  20. PORTB&=~_BV(SDA);
  21. asm("nop");
  22. PORTB&=~_BV(SCL);
  23. }
  24.  
  25. void stop_cond (void) //Генерация условия стоп
  26. {
  27. PORTB=_BV(SCL);
  28. asm("nop");
  29. PORTB&=~_BV(SDA);
  30. asm("nop");
  31. PORTB|=_BV(SDA);
  32. }
  33.  
  34. void send_byte (unsigned char data) //Отправка байта по I2С
  35. { unsigned char i;
  36. for (i=0;i<8;i++)
  37. {
  38. if ((data&0x80)==0x00) PORTB&=~_BV(SDA); //Выставить бит на SDA
  39. else PORTB|=_BV(SDA);
  40. asm("nop");
  41. PORTB|=_BV(SCL); //Записать его импульсом на SCL
  42. asm("nop");
  43. PORTB&=~_BV(SCL);
  44. data=data<<1;
  45. }
  46. DDRB&=~_BV(SDA);
  47. PORTB|=_BV(SDA);
  48.  
  49. asm("nop");
  50. PORTB|=_BV(SCL);
  51. asm("nop");
  52. if ((PINB&_BV(SDA))==_BV(SDA)) ack=1; //Считать ACK
  53. PORTB&=~_BV(SCL);
  54.  
  55. PORTB&=~_BV(SDA);
  56. DDRB|=_BV(SDA);
  57. }
  58.  
  59. unsigned char get_byte (void) //Считывание байта по I2C
  60. { unsigned char i, res=0;
  61.  
  62. DDRB&=~_BV(SDA);
  63. PORTB|=_BV(SDA);
  64.  
  65. for (i=0;i<8;i++)
  66. {
  67. res=res<<1;
  68. PORTB|=_BV(SCL); //Импульс на SCL
  69. asm("nop");
  70. if ((PINB&_BV(SDA))==_BV(SDA)) res=res|0x01;//Чтение SDA
  71. PORTB&=~_BV(SCL);
  72. asm("nop");
  73. }
  74.  
  75. PORTB&=~_BV(SDA); //Подтверждение ACK
  76. DDRB|=_BV(SDA);
  77.  
  78. asm("nop");
  79. PORTB|=_BV(SCL);
  80. asm("nop");
  81. PORTB&=~_BV(SCL);
  82.  
  83. return res;
  84. }
  85.  
  86. void read_date_time (void) //Считать данные в массив data
  87. { unsigned char i;
  88.  
  89. start_cond();
  90. send_byte (WRITE_PCF);
  91. send_byte (0x02);
  92.  
  93. start_cond();
  94. send_byte(READ_PCF);
  95. for (i=0;i<5;i++)
  96. data[i]=get_byte();
  97. stop_cond();
  98. }
  99.  
  100. void write_date_time (void) //Записать данные из массива data
  101. { unsigned char i;
  102.  
  103. start_cond();
  104. send_byte (WRITE_PCF);
  105. send_byte (0x02);
  106.  
  107. for (i=0;i<5;i++)
  108. send_byte (data[i]);
  109. stop_cond();
  110. }

Думаю особые объяснения не нужны, все делалось по диаграммам. Снятую осциллограмму с логического анализатора при чтении данных из PCF8583 можно глянуть здесь:

При отлове багов я столкнулся с «черной магией». Клавиши хаотично нажимались, не поддаваясь контролю. Причину их странного поведения выявить удалось не сразу. Виноватым оказалось пересечение проводков от клавиш и линии SCL, из которой передавались наводки, поэтому клавиши вынес на PORTC. Также мне показалось неудобной работа с BCD арифметикой и учетом всяких високосных/невисокосных годов, месяцев с 30/31 днями и т.д. Неудобен с точки зрения микроконтроллера человеческий календарь.
Посмотреть на работу часов можно здесь:

Скачать полный код проекта для AVR Studio 4

формате ВCD

Объясните начинающему, в каком виде хранятся данные (дата, время ...) в массиве data[]?
Если я правильно понимаю в формате ВCD?. Получается если число 16 то в микросхеме хранится в виде 0001 0110 а в массиве виде 0x16 (в шестнадцатеричной системе счисления)?

Все правильно, в программе в

Все правильно, в программе в архиве реализована перекодировка чисел вытянутых из PCF8583 в формате BCD в шестнадцатиричную систему. И наоборот при записи данных в микросхему.

Собрал конструкцию, все

Собрал конструкцию, все работает правда есть пару моментов:
1. На кране ЖКИ видно заметное мерцание подсветки.
2. При изменении частоты тактирования на внутренний источник 1МГц все только ухудшилось мигание стало еще более заметным.
3. Плохо работают кнопки, при входе в меню на экран выводится что-то непонятное.
4. На кране присутствует лишний символ "двоеточие" в начале второй строки.
5. Кварц выше 4МГц не советую ставить, так как начинает сбоить и сама шина I2C и ЖКИ.

1-2) Это мерцание

1-2) Это мерцание изображения, нужно увеличить частоту обновления.
3) Кнопки лучше перевесить на внешнее прерывания, все будет ок работать.
4)ХЗ кинь фотку.
5)Нужно переделать задержки, если хочешь другой кварц.

Купил батарейку CR2025 с

Купил батарейку CR2025 с напряжением 3,3Вольта вывел питание PCF8583 на нее, оставил подключенную только общую ногу.
Результат:

  1. пропало мигание на ЖКИ, видимо при работе от общего с микроконтроллером питания что-то происходило не хорошее.
  2. не так сильно чувствуется дребезг контактов при нажатии на кнопки

макетная плата на ней PCF8583, ATmega8, WH1602

Шина I2C и AVR пример, PCF8583 (RTC) вечный календарь

Согласно приведенному коду, при инициализации порта, пины (SDA, SCL) регистра DDRB настроены на выход, и записав в порт PORTB значение 0x0c мы тем самым садим шину на ноль. Не логичнее было бы при инициализации не занимать шину.

Логично, почему-то завтыкал с

Логично, почему-то завтыкал с 0х0с. Нужно 0х03. Вечером поправлю код.

Тогда порт нужно будет

Тогда порт нужно будет настроить на вход (с подтягивающими резисторами), так как если на линии будет присутствовать второй мастер (или еще чего), и вдруг он захочет начать вещание, то после прижатия линии к нулю может выйти из строя порт первого мастера.