Commençons par la HAL_I2C_Master_Transmit()
fonction. Si vous vérifiez sa déclaration:
HAL_StatusTypeDef HAL_I2C_Master_Transmit(I2C_HandleTypeDef *hi2c, uint16_t DevAddress, uint8_t *pData, uint16_t Size, uint32_t Timeout);
Problème mineur avec le 2ème paramètre, l'adresse du périphérique esclave. L'adresse du périphérique esclave est que b1010000
si nous la complétons au format 8 bits, elle le sera 0xA0
, comme vous l'avez dit. Maintenant, lorsque HAL_I2C_Master_Transmit()
vous passez cela, vous n'avez pas à définir le bit R / W manuellement, HAL le fera pour vous. Ainsi, lorsque vous appelez HAL_I2C_Master_Transmit()
le bit R / W transmis sera automatiquement 0 indiquant l'opération d'écriture et lorsque vous appelez HAL_I2C_Master_Receive()
le bit R / W transmis sera automatiquement 1 indiquant l'opération d'écriture . Vous avez mélangé les valeurs R / W mais je pense que c'est un peu pour la fonction, donc ce n'est pas une erreur réelle dans votre code.
Le 3ème paramètre ( uint8_t *pData
) est un pointeur vers un tampon qui contient les données à envoyer . Maintenant, dans votre appel, le troisième paramètre est 0x0C
celui de vos données réelles, l'adresse du registre. Le problème est qu'il sera interprété comme un pointeur (par le HAL_I2C_Master_Transmit()
) vers un emplacement mémoire, où des données non définies peuvent être trouvées.
Le 4ème paramètre est la taille du tampon , le nombre d'octets à envoyer. Si vous souhaitez envoyer un seul octet, ce paramètre doit être 1 et non 10.
I2C
Écrire des registres
Voici le schéma correspondant de la fiche technique.
Ainsi , après l' envoi de l'adresse esclave au bus, trois autres octets doivent être transmis: registre de pointeur , octet MSB , LSB octet . Une implémentation générale avec l'écriture HAL de registres 16 bits:
void write_register(uint8_t register_pointer, uint16_t register_value)
{
uint8_t data[3];
data[0] = register_pointer; // 0x0C in your example
data[1] = register_value>>8; // MSB byte of 16bit data
data[2] = register_value; // LSB byte of 16bit data
HAL_I2C_Master_Transmit(&hi2c1, 0xA0, data, 3, 100); // data is the start pointer of our array
}
Exemple avec vos valeurs: write_register(0x0C, 0x0054);
Alternativement, la fonction d'écriture de registre définie par HAL peut également être utilisée, qui a des paramètres supplémentaires pour transmettre l'adresse de registre et la taille d'adresse.
void write_register(uint8_t register_pointer, uint16_t register_value)
{
HAL_StatusTypeDef status = HAL_OK;
status = HAL_I2C_Mem_Write(&hi2c1, 0xA0, (uint16_t)register_pointer, I2C_MEMADD_SIZE_8BIT, (uint8_t*)(®ister_value), 2, 100);
/* Check the communication status */
if(status != HAL_OK)
{
// Error handling, for example re-initialization of the I2C peripheral
}
}
Maintenant, la HAL_I2C_Master_Receive()
fonction est presque la même que l'autre.
HAL_StatusTypeDef HAL_I2C_Master_Receive(I2C_HandleTypeDef *hi2c, uint16_t DevAddress, uint8_t *pData, uint16_t Size, uint32_t Timeout);
La seule différence est que le troisième paramètre est un pointeur vers le tampon où les données reçues seront stockées. C'est 0x02
dans votre code et je ne sais pas quel était votre but, mais il sera interprété comme un pointeur (malheureusement vers un emplacement mémoire aléatoire).
Lire les registres
I2CI2C
void read_register(uint8_t register_pointer, uint8_t* receive_buffer)
{
// first set the register pointer to the register wanted to be read
HAL_I2C_Master_Transmit(&hi2c1, 0xA0, ®ister_pointer, 1, 100); // note the & operator which gives us the address of the register_pointer variable
// receive the 2 x 8bit data into the receive buffer
HAL_I2C_Master_Receive(&hi2c1, 0xA0, receive_buffer, 2, 100);
}
Exemple:
uint8_t reg_ptr = 0x0C;
uint8_t buffer[2];
read_register(reg_ptr, buffer);
// the register content available in the buffer
Il existe également une fonction de lecture de registre définie par HAL, qui a.
uint16_t read_register(uint8_t register_pointer)
{
HAL_StatusTypeDef status = HAL_OK;
uint16_t return_value = 0;
status = HAL_I2C_Mem_Read(&hi2c1, 0xA0, (uint16_t)register_pointer, I2C_MEMADD_SIZE_8BIT, &return_value, 2, 100);
/* Check the communication status */
if(status != HAL_OK)
{
}
return return_value;
}
Lisez la section 8.5 Programmation de la fiche technique pour plus de détails.