Спасибо. Сделал, как написано, но все равно прочитать не могу - один плюс, что не зависает. Проходит только START и получение на него ACK, и больше ничего. Есть еще предположения где у меня траблы?
I2C_BUS->CR2 = I2C_CR2_START | 2<<16 | (address & 0xFE); /* Go */ while ((I2C_BUS->ISR & I2C_ISR_BUSY) == 0); // Ожидать выдачу старта // Сейчас либо I2C запросит первый байт для отправки, // Либо взлетит NACK-флаг, говорящий о том, что микросхема не отвечает. // Если взлетит NACK-флаг, отправку прекращаем.
while ((((I2C_BUS->ISR & I2C_ISR_TC)==0) && ((I2C_BUS->ISR & I2C_ISR_NACKF)==0)) && (I2C_BUS->ISR & I2C_ISR_BUSY)) { if (I2C_BUS->ISR & I2C_ISR_TXIS) I2C_BUS->TXDR = *reg_ptr++; // Отправляю адрес регистра }
// Принимаем байты до тех пор, пока не взлетит TC-флаг. // Если взлетит NACK-флаг, приём прекращаем. while ((((I2C_BUS->ISR & I2C_ISR_TC)==0) && ((I2C_BUS->ISR & I2C_ISR_NACKF)==0)) && (I2C_BUS->ISR & I2C_ISR_BUSY)) { if (I2C_BUS->ISR & I2C_ISR_RXNE) { *data++ = I2C_BUS->RXDR; // Принимаю данные Count++; } }
I2C_BUS->CR2 |= I2C_CR2_STOP; // Выдать стоп на шину while (I2C_BUS->ISR & I2C_ISR_BUSY); // Ожидать выдачу стопа // Очищаю флаги - необходимо для дальнейшей работы шины I2C_BUS->ICR |= I2C_ICR_STOPCF; // STOP флаг I2C_BUS->ICR |= I2C_ICR_NACKCF; // NACK флаг // Если есть ошибки на шине - очищаю флаги if (I2C_BUS->ISR & (I2C_ISR_ARLO | I2C_ISR_BERR)) { I2C_BUS->ICR |= I2C_ICR_ARLOCF; I2C_BUS->ICR |= I2C_ICR_BERRCF; }
if (Count == length) return I2C_SUCCESS; else return I2C_ERROR; }
_________________ А люди посмотрят и скажут: "Собаки летят. Вот и осень."
Обязательным условием долгой и стабильной работы Li-FePO4-аккумуляторов, в том числе и производства EVE Energy, является применение специализированных BMS-микросхем. Литий-железофосфатные АКБ отличаются такими характеристиками, как высокая многократность циклов заряда-разряда, безопасность, возможность быстрой зарядки, устойчивость к буферному режиму работы и приемлемая стоимость. Но для этих АКБ очень важен контроль процесса заряда и разряда для избегания воздействия внешнего зарядного напряжения после достижения 100% заряда. Инженеры КОМПЭЛ подготовили список таких решений от разных производителей.
Память AT24C08D 8Kbit...В даташит написано, что надо выдавать 8-битный адрес после СТАРТ: Accessing the device requires an 8-bit Device Address word following a Start condition to enable the device for a Read or Write operation. Since multiple slave devices can reside on the serial bus, each slave device must have its own unique address so the Master can access each device indepen dently.
Компания EVE выпустила новый аккумулятор серии PLM, сочетающий в себе высокую безопасность, длительный срок службы, широкий температурный диапазон и высокую токоотдачу даже при отрицательной температуре.
Эти аккумуляторы поддерживают заряд при температуре от -40/-20°С (сниженным значением тока), безопасны (не воспламеняются и не взрываются) при механическом повреждении (протыкание и сдавливание), устойчивы к вибрации. Они могут применяться как для автотранспорта (трекеры, маячки, сигнализация), так и для промышленных устройств мониторинга, IoT-устройств.
Эта... а вам компилятор варнинги не выдаёт? например на это: size << I2C_CR2_NBYTES_Pos. Справа у вас должно быть 16, а size описана как 8-бит. получается 0. Хотя если int у вашего компилятора 32 бита, то проблемы быть не должно...
Ладно, вот еще мой код для 8-битной адресации (но не проверенный. Не было у меня такой маленькой еепромки): Спойлер
unsigned char Count = 0; I2C_BUS->CR2 = I2C_CR2_START | 1<<16 |(address & 0xFE);/* Go */ while ((I2C_BUS->ISR & I2C_ISR_BUSY)== 0);// Ожидать выдачу старта // Сейчас либо I2C запросит первый байт для отправки, // Либо взлетит NACK-флаг, говорящий о том, что микросхема не отвечает. // Если взлетит NACK-флаг, отправку прекращаем. while ((((I2C_BUS->ISR & I2C_ISR_TC)==0)&&((I2C_BUS->ISR & I2C_ISR_NACKF)==0))&&(I2C_BUS->ISR & I2C_ISR_BUSY)){ if (I2C_BUS->ISR & I2C_ISR_TXIS) I2C_BUS->TXDR = reg_addr;// Отправляю адрес регистра } I2C_BUS->CR2 = I2C_CR2_START |((uint32_t)length<<16)| I2C_CR2_RD_WRN |(address & 0xFE);/* Restart */ while ((I2C_BUS->ISR & I2C_ISR_BUSY)==0);// Ожидать выдачу старта
// Принимаем байты до тех пор, пока не взлетит TC-флаг. // Если взлетит NACK-флаг, приём прекращаем. while ((((I2C_BUS->ISR & I2C_ISR_TC)==0)&&((I2C_BUS->ISR & I2C_ISR_NACKF)==0))&&(I2C_BUS->ISR & I2C_ISR_BUSY)){ if (I2C_BUS->ISR & I2C_ISR_RXNE){ *data++= I2C_BUS->RXDR;// Принимаю данные Count++; } }
I2C_BUS->CR2 |= I2C_CR2_STOP;// Выдать стоп на шину while (I2C_BUS->ISR & I2C_ISR_BUSY);// Ожидать выдачу стопа // Очищаю флаги - необходимо для дальнейшей работы шины I2C_BUS->ICR |= I2C_ICR_STOPCF;// STOP флаг I2C_BUS->ICR |= I2C_ICR_NACKCF;// NACK флаг // Если есть ошибки на шине - очищаю флаги if (I2C_BUS->ISR &(I2C_ISR_ARLO | I2C_ISR_BERR)){ I2C_BUS->ICR |= I2C_ICR_ARLOCF; I2C_BUS->ICR |= I2C_ICR_BERRCF; }
if (Count == length) return I2C_SUCCESS; return I2C_ERROR; }
_________________ А люди посмотрят и скажут: "Собаки летят. Вот и осень."
Нет варнингов не было, но подправил. Спасибо. Ваш код работает, т.е. идет чтение, но только не понятно я задаю чтение с адреса 0х00, а он каждый раз при чтении выдает разный адрес регистра. Если устройство перезапустить, то при каждом чтении опять по тем же адресам идет...
Добавлено after 14 minutes 54 seconds: Чтение идет в такой последовательности, при перезагрузке устройства (reg_addr = 0x00): 0x90, 0x08, 0x00, 0x20, 0xC5, 0x00, 0x00, 0x08, 0x4D...
хм.. тогда надо сравнить с 16-битной функцией. Только что попробовал в своём коде сделать, чтобы читало начальные установки с 1-го адреса - вижу все данные сместились. Так что rd_reg16 - вроде работает как следует. Хотя всё различие, что передаётся один или два байта адреса...
_________________ А люди посмотрят и скажут: "Собаки летят. Вот и осень."
Последний раз редактировалось uldemir Ср окт 04, 2017 14:49:09, всего редактировалось 1 раз.
viewtopic.php?p=3197984#p3197984 под спойлером. Имелось в виду i2c_rd_reg16 Попробуйте у себя сделать чтение не с 0 адреса, а с другого и посмотрите, какая тогда последовательность прочитанных байтов будет.
_________________ А люди посмотрят и скажут: "Собаки летят. Вот и осень."
но он там не сразу устанавливается, вроде. Ну, смеха для, перед передачей рестарта можно попробовать сформировать полный стоп. Для еепромки это должно быть без разницы - главное сделать фиктивную попытку записи по адресу, а потом оно с этого адреса и будет читать.
p.s. я в своей функции сделал счетчик переданных байт адреса и если он не равен 2 перед рестартом - выхожу с ошибкой. Но ошибка не появляется.
_________________ А люди посмотрят и скажут: "Собаки летят. Вот и осень."
сделал STOP теперь читает 0xFF постоянно. Буду дальше смотреть. Спасибо большое.
Добавлено after 40 minutes 33 seconds: uldemir, покажите пожалуйста настройку таймингов, т.к. я тайминги генерил через куб и при разных вариантах поведение одно и тоже. Может здесь собака порылась?!
ммм. я тайминги каким-то экселем считал, что с сайта производителя скачивал. У меня такая строчка: I2C_BUS->TIMINGR = (uint32_t)0x00100D14; /* (1) */ Тактируется от HSI.
Собственно, у меня самого сейчас с этим кодом возникли проблемы. Раз уж залез в этот девайс, решил добавить в меню еще одну настройку, в результате чего число параметров увеличилось и число записываемых байт перевалило размер страницы. Вот теперь что-то не получается сделать запись по-частям. Первую порцию записывает, а потом где-то виснет.
_________________ А люди посмотрят и скажут: "Собаки летят. Вот и осень."
Глючный на F1 , на F0 все работает прекрасно. Просто вы оба так и не въехали в сабж, хоть в RM все расписано от и до, и даже секвенции выложены. Учитесь читать...
Сейчас этот форум просматривают: нет зарегистрированных пользователей и гости: 33
Вы не можете начинать темы Вы не можете отвечать на сообщения Вы не можете редактировать свои сообщения Вы не можете удалять свои сообщения Вы не можете добавлять вложения