| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | // Copyright 2015-2016 Espressif Systems (Shanghai) PTE LTD
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Licensed under the Apache License, Version 2.0 (the "License");
 | 
					
						
							|  |  |  | // you may not use this file except in compliance with the License.
 | 
					
						
							|  |  |  | // You may obtain a copy of the License at
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //     http://www.apache.org/licenses/LICENSE-2.0
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Unless required by applicable law or agreed to in writing, software
 | 
					
						
							|  |  |  | // distributed under the License is distributed on an "AS IS" BASIS,
 | 
					
						
							|  |  |  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
					
						
							|  |  |  | // See the License for the specific language governing permissions and
 | 
					
						
							|  |  |  | // limitations under the License.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "esp32-hal-i2c.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  | #include "esp32-hal.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | #include "freertos/FreeRTOS.h"
 | 
					
						
							|  |  |  | #include "freertos/task.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  | #include "freertos/semphr.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | #include "rom/ets_sys.h"
 | 
					
						
							|  |  |  | #include "soc/i2c_reg.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  | #include "soc/i2c_struct.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | #include "soc/dport_reg.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  | //#define I2C_DEV(i)   (volatile i2c_dev_t *)((i)?DR_REG_I2C1_EXT_BASE:DR_REG_I2C_EXT_BASE)
 | 
					
						
							|  |  |  | //#define I2C_DEV(i)   ((i2c_dev_t *)(REG_I2C_BASE(i)))
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | #define I2C_SCL_IDX(p)  ((p==0)?I2CEXT0_SCL_OUT_IDX:((p==1)?I2CEXT1_SCL_OUT_IDX:0))
 | 
					
						
							|  |  |  | #define I2C_SDA_IDX(p) ((p==0)?I2CEXT0_SDA_OUT_IDX:((p==1)?I2CEXT1_SDA_OUT_IDX:0))
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct i2c_struct_t { | 
					
						
							|  |  |  |     i2c_dev_t * dev; | 
					
						
							| 
									
										
										
										
											2016-11-18 15:07:25 +02:00
										 |  |  | #if !CONFIG_DISABLE_HAL_LOCKS
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     xSemaphoreHandle lock; | 
					
						
							| 
									
										
										
										
											2016-11-18 15:07:25 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     uint8_t num; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum { | 
					
						
							|  |  |  |     I2C_CMD_RSTART, | 
					
						
							|  |  |  |     I2C_CMD_WRITE, | 
					
						
							|  |  |  |     I2C_CMD_READ, | 
					
						
							|  |  |  |     I2C_CMD_STOP, | 
					
						
							|  |  |  |     I2C_CMD_END | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-18 15:07:25 +02:00
										 |  |  | #if CONFIG_DISABLE_HAL_LOCKS
 | 
					
						
							|  |  |  | #define I2C_MUTEX_LOCK()
 | 
					
						
							|  |  |  | #define I2C_MUTEX_UNLOCK()
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static i2c_t _i2c_bus_array[2] = { | 
					
						
							|  |  |  |     {(volatile i2c_dev_t *)(DR_REG_I2C_EXT_BASE), 0}, | 
					
						
							|  |  |  |     {(volatile i2c_dev_t *)(DR_REG_I2C1_EXT_BASE), 1} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  | #define I2C_MUTEX_LOCK()    do {} while (xSemaphoreTake(i2c->lock, portMAX_DELAY) != pdPASS)
 | 
					
						
							|  |  |  | #define I2C_MUTEX_UNLOCK()  xSemaphoreGive(i2c->lock)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static i2c_t _i2c_bus_array[2] = { | 
					
						
							|  |  |  |     {(volatile i2c_dev_t *)(DR_REG_I2C_EXT_BASE), NULL, 0}, | 
					
						
							|  |  |  |     {(volatile i2c_dev_t *)(DR_REG_I2C1_EXT_BASE), NULL, 1} | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2016-11-18 15:07:25 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  | i2c_err_t i2cAttachSCL(i2c_t * i2c, int8_t scl) | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     if(i2c == NULL){ | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |         return I2C_ERROR_DEV; | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-14 13:11:10 +02:00
										 |  |  |     pinMode(scl, OUTPUT_OPEN_DRAIN | PULLUP); | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     pinMatrixOutAttach(scl, I2C_SCL_IDX(i2c->num), false, false); | 
					
						
							|  |  |  |     pinMatrixInAttach(scl, I2C_SCL_IDX(i2c->num), false); | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |     return I2C_ERROR_OK; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  | i2c_err_t i2cDetachSCL(i2c_t * i2c, int8_t scl) | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     if(i2c == NULL){ | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |         return I2C_ERROR_DEV; | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     pinMatrixOutDetach(scl, false, false); | 
					
						
							|  |  |  |     pinMatrixInDetach(I2C_SCL_IDX(i2c->num), false, false); | 
					
						
							|  |  |  |     pinMode(scl, INPUT); | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |     return I2C_ERROR_OK; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  | i2c_err_t i2cAttachSDA(i2c_t * i2c, int8_t sda) | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     if(i2c == NULL){ | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |         return I2C_ERROR_DEV; | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-14 13:11:10 +02:00
										 |  |  |     pinMode(sda, OUTPUT_OPEN_DRAIN | PULLUP); | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     pinMatrixOutAttach(sda, I2C_SDA_IDX(i2c->num), false, false); | 
					
						
							|  |  |  |     pinMatrixInAttach(sda, I2C_SDA_IDX(i2c->num), false); | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |     return I2C_ERROR_OK; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  | i2c_err_t i2cDetachSDA(i2c_t * i2c, int8_t sda) | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     if(i2c == NULL){ | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |         return I2C_ERROR_DEV; | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     pinMatrixOutDetach(sda, false, false); | 
					
						
							|  |  |  |     pinMatrixInDetach(I2C_SDA_IDX(i2c->num), false, false); | 
					
						
							|  |  |  |     pinMode(sda, INPUT); | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |     return I2C_ERROR_OK; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * index     - command index (0 to 15) | 
					
						
							|  |  |  |  * op_code   - is the command | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |  * byte_num  - This register is to store the amounts of data that is read and written. byte_num in RSTART, STOP, END is null. | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |  * ack_val   - Each data byte is terminated by an ACK bit used to set the bit level. | 
					
						
							|  |  |  |  * ack_exp   - This bit is to set an expected ACK value for the transmitter. | 
					
						
							|  |  |  |  * ack_check - This bit is to decide whether the transmitter checks ACK bit. 1 means yes and 0 means no. | 
					
						
							|  |  |  |  * */ | 
					
						
							|  |  |  | void i2cSetCmd(i2c_t * i2c, uint8_t index, uint8_t op_code, uint8_t byte_num, bool ack_val, bool ack_exp, bool ack_check) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     i2c->dev->command[index].val = 0; | 
					
						
							|  |  |  |     i2c->dev->command[index].ack_en = ack_check; | 
					
						
							|  |  |  |     i2c->dev->command[index].ack_exp = ack_exp; | 
					
						
							|  |  |  |     i2c->dev->command[index].ack_val = ack_val; | 
					
						
							|  |  |  |     i2c->dev->command[index].byte_num = byte_num; | 
					
						
							|  |  |  |     i2c->dev->command[index].op_code = op_code; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-17 00:46:56 +02:00
										 |  |  | void i2cResetCmd(i2c_t * i2c){ | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  |     for(i=0;i<16;i++){ | 
					
						
							|  |  |  |         i2c->dev->command[i].val = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  | void i2cResetFiFo(i2c_t * i2c) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     i2c->dev->fifo_conf.tx_fifo_rst = 1; | 
					
						
							|  |  |  |     i2c->dev->fifo_conf.tx_fifo_rst = 0; | 
					
						
							|  |  |  |     i2c->dev->fifo_conf.rx_fifo_rst = 1; | 
					
						
							|  |  |  |     i2c->dev->fifo_conf.rx_fifo_rst = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  | i2c_err_t i2cWrite(i2c_t * i2c, uint16_t address, bool addr_10bit, uint8_t * data, uint8_t len, bool sendStop) | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |     int i; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     uint8_t index = 0; | 
					
						
							|  |  |  |     uint8_t dataLen = len + (addr_10bit?2:1); | 
					
						
							|  |  |  |     address = (address << 1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     if(i2c == NULL){ | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |         return I2C_ERROR_DEV; | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     I2C_MUTEX_LOCK(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     while(dataLen) { | 
					
						
							|  |  |  |         uint8_t willSend = (dataLen > 32)?32:dataLen; | 
					
						
							|  |  |  |         uint8_t dataSend = willSend; | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-10 14:54:36 +03:00
										 |  |  |         i2cResetFiFo(i2c); | 
					
						
							| 
									
										
										
										
											2016-11-17 00:46:56 +02:00
										 |  |  |         i2cResetCmd(i2c); | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |         //Clear Interrupts
 | 
					
						
							|  |  |  |         i2c->dev->int_clr.val = 0xFFFFFFFF; | 
					
						
							| 
									
										
										
										
											2016-10-10 14:54:36 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |         //CMD START
 | 
					
						
							|  |  |  |         i2cSetCmd(i2c, 0, I2C_CMD_RSTART, 0, false, false, false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         //CMD WRITE(ADDRESS + DATA)
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         if(!index) { | 
					
						
							|  |  |  |             i2c->dev->fifo_data.data = address & 0xFF; | 
					
						
							|  |  |  |             dataSend--; | 
					
						
							|  |  |  |             if(addr_10bit) { | 
					
						
							|  |  |  |                 i2c->dev->fifo_data.data = (address >> 8) & 0xFF; | 
					
						
							|  |  |  |                 dataSend--; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |         i = 0; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         while(i<dataSend) { | 
					
						
							|  |  |  |             i++; | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |             i2c->dev->fifo_data.val = data[index++]; | 
					
						
							|  |  |  |             while(i2c->dev->status_reg.tx_fifo_cnt < i); | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |         i2cSetCmd(i2c, 1, I2C_CMD_WRITE, willSend, false, false, true); | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         dataLen -= willSend; | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         //CMD STOP or CMD END if there is more data
 | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |         if(dataLen || !sendStop) { | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |             i2cSetCmd(i2c, 2, I2C_CMD_END, 0, false, false, false); | 
					
						
							|  |  |  |         } else if(sendStop) { | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |             i2cSetCmd(i2c, 2, I2C_CMD_STOP, 0, false, false, false); | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         //START Transmission
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         i2c->dev->ctr.trans_start = 1; | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         //WAIT Transmission
 | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |         uint32_t startAt = millis(); | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |         while(1) { | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |             //have been looping for too long
 | 
					
						
							|  |  |  |             if((millis() - startAt)>50){ | 
					
						
							|  |  |  |                 //log_e("Timeout! Addr: %x", address >> 1);
 | 
					
						
							|  |  |  |                 I2C_MUTEX_UNLOCK(); | 
					
						
							|  |  |  |                 return I2C_ERROR_BUS; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-10 14:54:36 +03:00
										 |  |  |             //Bus failed (maybe check for this while waiting?
 | 
					
						
							|  |  |  |             if(i2c->dev->int_raw.arbitration_lost) { | 
					
						
							|  |  |  |                 //log_e("Bus Fail! Addr: %x", address >> 1);
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |                 I2C_MUTEX_UNLOCK(); | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |                 return I2C_ERROR_BUS; | 
					
						
							| 
									
										
										
										
											2016-10-10 14:54:36 +03:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             //Bus timeout
 | 
					
						
							|  |  |  |             if(i2c->dev->int_raw.time_out) { | 
					
						
							|  |  |  |                 //log_e("Bus Timeout! Addr: %x", address >> 1);
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |                 I2C_MUTEX_UNLOCK(); | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |                 return I2C_ERROR_TIMEOUT; | 
					
						
							| 
									
										
										
										
											2016-10-10 14:54:36 +03:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             //Transmission did not finish and ACK_ERR is set
 | 
					
						
							|  |  |  |             if(i2c->dev->int_raw.ack_err) { | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |                 //log_w("Ack Error! Addr: %x", address >> 1);
 | 
					
						
							| 
									
										
										
										
											2016-12-15 01:41:54 +02:00
										 |  |  |                 while(i2c->dev->status_reg.bus_busy); | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |                 I2C_MUTEX_UNLOCK(); | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |                 return I2C_ERROR_ACK; | 
					
						
							| 
									
										
										
										
											2016-10-10 14:54:36 +03:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |             if((sendStop && i2c->dev->command[2].done) || !i2c->dev->status_reg.bus_busy){ | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     I2C_MUTEX_UNLOCK(); | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |     return I2C_ERROR_OK; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  | i2c_err_t i2cRead(i2c_t * i2c, uint16_t address, bool addr_10bit, uint8_t * data, uint8_t len, bool sendStop) | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | { | 
					
						
							|  |  |  |     address = (address << 1) | 1; | 
					
						
							|  |  |  |     uint8_t addrLen = (addr_10bit?2:1); | 
					
						
							|  |  |  |     uint8_t index = 0; | 
					
						
							|  |  |  |     uint8_t cmdIdx; | 
					
						
							|  |  |  |     uint8_t willRead; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     if(i2c == NULL){ | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |         return I2C_ERROR_DEV; | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     I2C_MUTEX_LOCK(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-10 14:54:36 +03:00
										 |  |  |     i2cResetFiFo(i2c); | 
					
						
							| 
									
										
										
										
											2016-11-17 00:46:56 +02:00
										 |  |  |     i2cResetCmd(i2c); | 
					
						
							| 
									
										
										
										
											2016-10-10 14:54:36 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |     //CMD START
 | 
					
						
							|  |  |  |     i2cSetCmd(i2c, 0, I2C_CMD_RSTART, 0, false, false, false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     //CMD WRITE ADDRESS
 | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |     i2c->dev->fifo_data.val = address & 0xFF; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     if(addr_10bit) { | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |         i2c->dev->fifo_data.val = (address >> 8) & 0xFF; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |     i2cSetCmd(i2c, 1, I2C_CMD_WRITE, addrLen, false, false, true); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     while(len) { | 
					
						
							|  |  |  |         cmdIdx = (index)?0:2; | 
					
						
							|  |  |  |         willRead = (len > 32)?32:(len-1); | 
					
						
							| 
									
										
										
										
											2016-10-10 14:54:36 +03:00
										 |  |  |         if(cmdIdx){ | 
					
						
							|  |  |  |             i2cResetFiFo(i2c); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-14 18:43:32 +02:00
										 |  |  |         if(willRead){ | 
					
						
							|  |  |  |             i2cSetCmd(i2c, cmdIdx++, I2C_CMD_READ, willRead, false, false, false); | 
					
						
							| 
									
										
										
										
											2016-12-14 23:31:28 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if((len - willRead) > 1) { | 
					
						
							|  |  |  |             i2cSetCmd(i2c, cmdIdx++, I2C_CMD_END, 0, false, false, false); | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             willRead++; | 
					
						
							|  |  |  |             i2cSetCmd(i2c, cmdIdx++, I2C_CMD_READ, 1, true, false, false); | 
					
						
							|  |  |  |             if(sendStop) { | 
					
						
							|  |  |  |                 i2cSetCmd(i2c, cmdIdx++, I2C_CMD_STOP, 0, false, false, false); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |         //Clear Interrupts
 | 
					
						
							|  |  |  |         i2c->dev->int_clr.val = 0xFFFFFFFF; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         //START Transmission
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         i2c->dev->ctr.trans_start = 1; | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         //WAIT Transmission
 | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |         uint32_t startAt = millis(); | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |         while(1) { | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |             //have been looping for too long
 | 
					
						
							|  |  |  |             if((millis() - startAt)>50){ | 
					
						
							|  |  |  |                 //log_e("Timeout! Addr: %x", address >> 1);
 | 
					
						
							|  |  |  |                 I2C_MUTEX_UNLOCK(); | 
					
						
							|  |  |  |                 return I2C_ERROR_BUS; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-10 14:54:36 +03:00
										 |  |  |             //Bus failed (maybe check for this while waiting?
 | 
					
						
							|  |  |  |             if(i2c->dev->int_raw.arbitration_lost) { | 
					
						
							|  |  |  |                 //log_e("Bus Fail! Addr: %x", address >> 1);
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |                 I2C_MUTEX_UNLOCK(); | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |                 return I2C_ERROR_BUS; | 
					
						
							| 
									
										
										
										
											2016-10-10 14:54:36 +03:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             //Bus timeout
 | 
					
						
							|  |  |  |             if(i2c->dev->int_raw.time_out) { | 
					
						
							|  |  |  |                 //log_e("Bus Timeout! Addr: %x", address >> 1);
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |                 I2C_MUTEX_UNLOCK(); | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |                 return I2C_ERROR_TIMEOUT; | 
					
						
							| 
									
										
										
										
											2016-10-10 14:54:36 +03:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             //Transmission did not finish and ACK_ERR is set
 | 
					
						
							|  |  |  |             if(i2c->dev->int_raw.ack_err) { | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |                 //log_w("Ack Error! Addr: %x", address >> 1);
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |                 I2C_MUTEX_UNLOCK(); | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |                 return I2C_ERROR_ACK; | 
					
						
							| 
									
										
										
										
											2016-10-10 14:54:36 +03:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if(i2c->dev->command[cmdIdx-1].done) { | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         int i = 0; | 
					
						
							|  |  |  |         while(i<willRead) { | 
					
						
							|  |  |  |             i++; | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |             data[index++] = i2c->dev->fifo_data.val & 0xFF; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         } | 
					
						
							|  |  |  |         len -= willRead; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     I2C_MUTEX_UNLOCK(); | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |     return I2C_ERROR_OK; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  | i2c_err_t i2cSetFrequency(i2c_t * i2c, uint32_t clk_speed) | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     if(i2c == NULL){ | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |         return I2C_ERROR_DEV; | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |     uint32_t period = (APB_CLK_FREQ/clk_speed) / 2; | 
					
						
							|  |  |  |     uint32_t halfPeriod = period/2; | 
					
						
							|  |  |  |     uint32_t quarterPeriod = period/4; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     I2C_MUTEX_LOCK(); | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |     //the clock num during SCL is low level
 | 
					
						
							| 
									
										
										
										
											2017-01-05 13:54:40 +02:00
										 |  |  |     i2c->dev->scl_low_period.period = period; | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |     //the clock num during SCL is high level
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     i2c->dev->scl_high_period.period = period; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |     //the clock num between the negedge of SDA and negedge of SCL for start mark
 | 
					
						
							|  |  |  |     i2c->dev->scl_start_hold.time = halfPeriod; | 
					
						
							|  |  |  |     //the clock num between the posedge of SCL and the negedge of SDA for restart mark
 | 
					
						
							|  |  |  |     i2c->dev->scl_rstart_setup.time = halfPeriod; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |     //the clock num after the STOP bit's posedge
 | 
					
						
							|  |  |  |     i2c->dev->scl_stop_hold.time = halfPeriod; | 
					
						
							|  |  |  |     //the clock num between the posedge of SCL and the posedge of SDA
 | 
					
						
							|  |  |  |     i2c->dev->scl_stop_setup.time = halfPeriod; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |     //the clock num I2C used to hold the data after the negedge of SCL.
 | 
					
						
							|  |  |  |     i2c->dev->sda_hold.time = quarterPeriod; | 
					
						
							|  |  |  |     //the clock num I2C used to sample data on SDA after the posedge of SCL
 | 
					
						
							|  |  |  |     i2c->dev->sda_sample.time = quarterPeriod; | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     I2C_MUTEX_UNLOCK(); | 
					
						
							| 
									
										
										
										
											2016-10-11 15:11:51 +03:00
										 |  |  |     return I2C_ERROR_OK; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | uint32_t i2cGetFrequency(i2c_t * i2c) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     if(i2c == NULL){ | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-05 13:54:40 +02:00
										 |  |  |     return APB_CLK_FREQ/(i2c->dev->scl_low_period.period+i2c->dev->scl_high_period.period); | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * mode          - 0 = Slave, 1 = Master | 
					
						
							|  |  |  |  * slave_addr    - I2C Address | 
					
						
							|  |  |  |  * addr_10bit_en - enable slave 10bit address mode. | 
					
						
							|  |  |  |  * */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | i2c_t * i2cInit(uint8_t i2c_num, uint16_t slave_addr, bool addr_10bit_en) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     if(i2c_num > 1){ | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     i2c_t * i2c = &_i2c_bus_array[i2c_num]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-18 15:07:25 +02:00
										 |  |  | #if !CONFIG_DISABLE_HAL_LOCKS
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     if(i2c->lock == NULL){ | 
					
						
							|  |  |  |         i2c->lock = xSemaphoreCreateMutex(); | 
					
						
							|  |  |  |         if(i2c->lock == NULL) { | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-11-18 15:07:25 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     if(i2c_num == 0) { | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         SET_PERI_REG_MASK(DPORT_PERIP_CLK_EN_REG,DPORT_I2C_EXT0_CLK_EN); | 
					
						
							|  |  |  |         CLEAR_PERI_REG_MASK(DPORT_PERIP_RST_EN_REG,DPORT_I2C_EXT0_RST); | 
					
						
							| 
									
										
										
										
											2016-10-11 14:20:02 +03:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |         SET_PERI_REG_MASK(DPORT_PERIP_CLK_EN_REG,DPORT_I2C_EXT1_CLK_EN); | 
					
						
							|  |  |  |         CLEAR_PERI_REG_MASK(DPORT_PERIP_RST_EN_REG,DPORT_I2C_EXT1_RST); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-10-14 03:24:20 +03:00
										 |  |  |      | 
					
						
							|  |  |  |     I2C_MUTEX_LOCK(); | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |     i2c->dev->ctr.val = 0; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     i2c->dev->ctr.ms_mode = (slave_addr == 0); | 
					
						
							|  |  |  |     i2c->dev->ctr.sda_force_out = 1 ; | 
					
						
							|  |  |  |     i2c->dev->ctr.scl_force_out = 1 ; | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |     i2c->dev->ctr.clk_en = 1; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |     //the max clock number of receiving  a data
 | 
					
						
							| 
									
										
										
										
											2016-12-09 12:51:36 +02:00
										 |  |  |     i2c->dev->timeout.tout = 400000;//clocks max=1048575
 | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  |     //disable apb nonfifo access
 | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     i2c->dev->fifo_conf.nonfifo_en = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-08 17:32:11 +03:00
										 |  |  |     i2c->dev->slave_addr.val = 0; | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  |     if (slave_addr) { | 
					
						
							|  |  |  |         i2c->dev->slave_addr.addr = slave_addr; | 
					
						
							|  |  |  |         i2c->dev->slave_addr.en_10bit = addr_10bit_en; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-10-14 03:24:20 +03:00
										 |  |  |     I2C_MUTEX_UNLOCK(); | 
					
						
							| 
									
										
										
										
											2016-10-06 14:21:30 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return i2c; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-14 02:41:22 +02:00
										 |  |  | void i2cInitFix(i2c_t * i2c){ | 
					
						
							|  |  |  |     if(i2c == NULL){ | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     I2C_MUTEX_LOCK(); | 
					
						
							|  |  |  |     i2cResetFiFo(i2c); | 
					
						
							|  |  |  |     i2cResetCmd(i2c); | 
					
						
							|  |  |  |     i2c->dev->int_clr.val = 0xFFFFFFFF; | 
					
						
							|  |  |  |     i2cSetCmd(i2c, 0, I2C_CMD_RSTART, 0, false, false, false); | 
					
						
							|  |  |  |     i2c->dev->fifo_data.data = 0; | 
					
						
							|  |  |  |     i2cSetCmd(i2c, 1, I2C_CMD_WRITE, 1, false, false, false); | 
					
						
							|  |  |  |     i2cSetCmd(i2c, 2, I2C_CMD_STOP, 0, false, false, false); | 
					
						
							|  |  |  |     i2c->dev->ctr.trans_start = 1; | 
					
						
							|  |  |  |     while(!i2c->dev->command[2].done); | 
					
						
							|  |  |  |     I2C_MUTEX_UNLOCK(); | 
					
						
							|  |  |  | } |