[Из песочницы] Особенности RTC M41T56

habr.png

M41T56 это микросхема Real Time Clock, являющаяся аналогом популярной DS1307. И хотя даже цоколевка микросхем совпадает, у них есть существенные отличия, о которых я постараюсь рассказать.

Краткое описание


Я не буду останавливаться на работе шины IIC, отмечу только что обе микросхемы имеет адрес 0xd0. Для работы с временем микросхемы содержат семь регистров счета и регистр управления. Регистры счета содержат числа в двоично-десятичном формате, однако некоторые биты имеют особое значение.

Регистры
xxx — значение бита не определено.


Биты
x — после включения состояние бита может быть любым.
Различия начинаются в назначении битов 7, 6 и 5 регистра часов. В M41T56 биты 7 и 6 используются для индикации перехода в новое столетие, а биты 5 и 4 используются для счета десятков часов. Причем счет часов возможен только в режиме 24, режим AM/PM недоступен. В DS1307 бит 7 не используется, ноль в бите 6 указывает, что используется режим счета 24 и этом случае биты 5 и 4 содержат десятки часов. Если в бите 6 единица, то бит 5 становится флагом AM/PM, а бит 4 содержит десятки часов.

Существенные отличия есть в регистре управления, которой содержит слово коррекции хода.

Коррекция хода


M41T56 позволяет компенсировать погрешность кварцевого резонатора в диапазоне от -62 до +124 ppm, что дает отклонение не более чем ±5 секунд за месяц. За компенсацию отвечают шесть младших битов управляющего регистра. Биты 4–0 содержат целое число без знака величины коррекции, а бит 5 задает направление коррекции. Если бит 5 содержит ноль, то ход замедляется с шагом 2.034 ppm, в противном случае RTC ускоряется с шагом 4.068. Это неудобно, поэтому я набросал пару простых функций для преобразования из ppm в слово коррекции и обратно.

/* Маска для получения только значения калибровки без знака. */
#define MASK_CALIBR             ((1 << 4) | (1 << 3) | (1 << 2) | (1 << 1) | (1 << 0))
/* Маска для получения только значения калибровки без знака. */
#define MASK_CALIBR_SIGN        (1 << 5)

/**
 * Получение значения коррекции хода.
 * @param caliber Калибровочное слово из RTC.
 * @return Коррекция хода в ppm.
 */
int8_t caliber_to_ppm(uint8_t caliber)
{
        int8_t result = caliber & MASK_CALIBR;

        result = (uint8_t) result * 2;
        if ((caliber & MASK_CALIBR_SIGN) != 0) {
                result = -result;
        } else {
                result = (uint8_t) result * 2;
        }

        return result;
}

/**
 * Получение калибровочного слова.
 * @param ppm Коррекция хода в ppm.
 * @return Калибровочное слово для RTC.
 */
uint8_t ppm_to_caliber(int8_t ppm)
{
        uint8_t result;

        if (ppm < 0) {
                result = (uint8_t) (-ppm + 1) / 2;
                result |= MASK_CALIBR_SIGN;
        } else {
                result = (uint8_t) (ppm + 2) / 4;
        }

        return result;
}


Детектирование сбоев


Ни DS1307, но M41T56 не умеют детектировать сбои генерации, но гарантируют, что при включении питания некоторые биты будут в определенном состоянии. M41T56 при включении в управляющем регистре будет 10xxxxxx. Для отслеживания сбоев в программе можно придерживаться следующего алгоритма. Если при включении микроконтроллера регистр управления RTC содержит 10xxxxxx, значит был сбой питания и в регистр нужно записать значение, старшие биты которого не равны 10. Самое простое — записать слово коррекции с битами 7 и 6 равными нулю.

Литература


  • Справочный листок M41T56
  • Справочный листок DS1307

© Habrahabr.ru