Project

General

Profile

Работа с I2C master (Си)

Рассмотрим пример работы с I2C0. Для быстрой проверки работоспособности добавим в пример вывод полученного значения по UART.
Вывод по UART будем использовать для платы HW1-MCp. Добавление вывода значения полученных данных по I2C0 для платы LDM-MCp
показано в разделе "Работа с UART (Си)".

API функции I2C0:

Кроме полезных макросов имеются функции для вывода данных по I2C.
Разберём подробнее каждую функцию I2C0, которая объявлена в i2c.h

1)void i2c0_init(char enable, int clock_speed);

Функция i2c0_init() выполняет инициализацию I2C0.
enable - разрешение работы I2C0
clock_speed - скорость работы I2C0

2)void i2c0_send(char byte, char slv_addr, int addr);

Функция i2c0_send() выполняет посылку байта на ведомое устройство.
byte - байт для отправки по I2C0
slv_addr - адрес ведомого устройства I2C0
addr - внутренний адрес ведомого устройства I2C0 (2 байта)

3)i2c0_get(char slv_addr, int addr);

Функция i2c0_get() получает байт от ведомого устройства.
slv_addr - адрес ведомого устройства I2C0
addr - внутренний адрес ведомого устройства I2C0 (2 байта)

Замечание: Функции i2c0_send() и i2c0_get() охватывают только устройства
с 7-ми битной адресацией и внутренним адресом в 2 байта. Написание функций под
конкретное устройство разбирается ниже.

Передача и приём байта по I2C0

На отладочных платах LDM-MCp и HW1-MCp к интерфейсу I2C0 подключена память.
Рассмотрим пример записи байта по нулевому адресу в память с адресом 0xA0.

1) Сконфигурируем UART0 и отправим байт

    int slv_addr;
    char byte_user;
    char byte;
    UART_InitTypeDef UART_InitStructure;

    UART_InitStructure.BaudRate = 38400;
    UART_InitStructure.TypeParity = 0x00000000;
    UART_InitStructure.Parity = 0x00000000;
    UART_InitStructure.FlowControl = 0x00000000; 
    UART_InitStructure.Mode = 0x00000003; /

    GPIOB->BPS = 0x00000300; 
    uart_init(UART0, &UART_InitStructure);
    UART_SEND_BYTE(0xAB, UART0);
    GPIOA->BPS = 0x00C00000; // разрешаем альтернативные функции порта для i2c0(master)

2) Ожидаем байт от пользователя по UART0

    slv_addr = 0xA0; // задаём адрес ведомого устройства по I2C0
    while(UART_NEW_DATA(UART0) == 0); // ожидаем байт от пользователя по UART0
    byte_user = UART_GET_BYTE(UART0); // получаем байт по UART0

3) Инициализация I2C0

Разрешаем работы I2C0, а также задаём скорость обмена данными 100 Кбит/с.

i2c0_init(1, 100000);

4) Передача байта по I2C0

i2c0_send(byte_user, slv_addr, 0);

5) Приём байта по I2C0

byte = i2c0_get(slv_addr, 0);

6) Передача полученного байта по UART0

UART_SEND_BYTE(0xAB, UART0);

Функции приёма и передачи под определённое устройство

Рассмотрим формирование функций передачи и приёма.
Функция передачи

 i2c0_send (byte_user, slv_addr, addr);

Может быть реализована следующим образом на Си:

    I2C0_TX_DATA(slv_addr); // установка адреса ведомого устройства
    I2C0_CMD(I2C0_START_WR); // посылка START + WR (СТАРТ и запись адреса)
    while(I2C0_TIP == 1); // ожидание передачи данных
    while(I2C0_ACK_RX == 1); // ожидание ACK

    I2C0_TX_DATA(0x00); // установка первого байта на передачу (внутренний адрес устройства)
    I2C0_CMD(I2C0_WR); //посылка WR
    while(I2C0_TIP == 1); // ожидание передачи данных
    while(I2C0_ACK_RX == 1); // ожидание ACK

    I2C0_TX_DATA(0x00); // установка второго байта на передачу (внутренний адрес устройства)
    I2C0_CMD(I2C0_WR); // посылка WR
    while(I2C0_TIP == 1); // ожидание передачи данных
    while(I2C0_ACK_RX == 1); // ожидание ACK

    I2C0_TX_DATA(byte_user); // установка байта на передачу
    I2C0_CMD(I2C0_WR); // посылка WR
    while(I2C0_TIP == 1); // ожидание передачи данных
    while(I2C0_ACK_RX == 1); // ожидание ACK

    I2C0_CMD(I2C0_STOP_WR); //посылка STOP + WR
    while(I2C0_TIP == 1); // ожидание передачи данных
    while(I2C0_ACK_RX == 1); // ожидание ACK

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