ESP32 IDF 讀取雙路AS5600

白了少年頭,空悲切。
又有哥們生孩子了。他們有車有房有老婆還有孩子,也不知咋掙來的。搞得我聚會都不想去了。
鬱悶。
沒辦法,自己只有努力。
胸藏萬卷憑吞吐,筆有千鈞任翕張。

前言

  • 之前用ESP32驅動兩路AS5600沒有成功,後面簡單梳理了下IIC的邏輯,才反應過來。下面記錄一下。

ESP32 IDF環境下雙路IIC的切換讀取

在官方提供的幾個IIC的demo中,都是不做切換的讀取。按照例程的步驟初始化單路IIC,讀取時沒問題。多路切換時,只需要注意切換 i2c_master_port 這個參數就行了。

首先,需要在初始化中分別註冊兩路IIC及其對應的引腳:

 esp_err_t i2c_master_init(void)
{

	 int i2c_master_port = I2C_AS5600_NUM0;  //這個port是切換IIC的關鍵,在i2c_driver_install中要註冊不同的port

	     i2c_config_t conf = {
	         .mode = I2C_MODE_MASTER,
	         .sda_io_num = I2C_AS5600_SDA_0,
	         .sda_pullup_en = GPIO_PULLUP_ENABLE,
	 		.scl_io_num = I2C_AS5600_SCL_0,
	         .scl_pullup_en = GPIO_PULLUP_ENABLE,
	         .master.clk_speed = I2C_AS5600_FREQ_HZ,
	     };

	     i2c_param_config(i2c_master_port, &conf);

	     i2c_driver_install(i2c_master_port, conf.mode, I2C_AS5600_RX_BUF_DISABLE, I2C_AS5600_TX_BUF_DISABLE, 0);

	     i2c_master_port = I2C_AS5600_NUM1;//另一路IIC

	     i2c_config_t conf1 = {
	         .mode = I2C_MODE_MASTER,
	         .sda_io_num = I2C_AS5600_SDA_1,
	         .sda_pullup_en = GPIO_PULLUP_ENABLE,
	 		.scl_io_num = I2C_AS5600_SCL_1,
	         .scl_pullup_en = GPIO_PULLUP_ENABLE,
	         .master.clk_speed = I2C_AS5600_FREQ_HZ,
	     };

	     i2c_param_config(i2c_master_port, &conf1);

	     return i2c_driver_install(i2c_master_port, conf1.mode, I2C_AS5600_RX_BUF_DISABLE, I2C_AS5600_TX_BUF_DISABLE, 0);


}

然後在使用過程中,切換 i2c_master_port 參數就行了:


static  esp_err_t i2c_master_DataGet(uint8_t sensorNum,float *returnAngel){

	uint8_t angle_high = 0;
	    uint8_t angle_low = 0;
	    uint16_t result = 0;
	    float angle=0;
	    i2c_cmd_handle_t cmd;
	    int i2c_master_port = I2C_AS5600_NUM0;

	    if(sensorNum == 0){
	    	 i2c_master_port = I2C_AS5600_NUM0;//這個port是切換IIC的關鍵,讀取時使用不同的port就能切換IIC
	    }else{
	    	i2c_master_port = I2C_AS5600_NUM1;
	    }
	    //讀取HighAddr
	    cmd = i2c_cmd_link_create();
	    i2c_master_start(cmd);
	    i2c_master_write_byte(cmd, I2C_AS5600_SENSOR_ADDR << 1 | WRITE_BIT, ACK_CHECK_EN);
	    i2c_master_write_byte(cmd, I2C_AS5600_ANGLE_ADDRH, ACK_CHECK_EN);
	    i2c_master_stop(cmd);
	    i2c_master_cmd_begin(i2c_master_port, cmd, 10 / portTICK_PERIOD_MS);  //可以看到,上面的操作應該是需要這個begin開始啓動,使用port來確認對哪個IIC進行操作
	    i2c_cmd_link_delete(cmd);
	    //vTaskDelay(1 / portTICK_PERIOD_MS);

	    cmd = i2c_cmd_link_create();
	    i2c_master_start(cmd);
	    i2c_master_write_byte(cmd, I2C_AS5600_SENSOR_ADDR << 1 | READ_BIT, ACK_CHECK_EN);
	    i2c_master_read_byte(cmd, &angle_high, ACK_VAL); //0x0c是高位
	    i2c_master_read_byte(cmd, &angle_low, NACK_VAL);
	    i2c_master_stop(cmd);
	    i2c_master_cmd_begin(i2c_master_port, cmd, 10 / portTICK_PERIOD_MS);
	    i2c_cmd_link_delete(cmd);

	    result=(uint16_t)(angle_high<<8|angle_low); //一共就11位 注意
	    angle=((int) result & 0b0000111111111111)*360.0/4096.0;
	    *returnAngel = angle;
	    //printf("angle%d: %.2f \n",sensorNum,angle);

return 0;
}

可以看到,IDF環境對於IIC的處理是將IIC的操作放到了 i2c_master_start()和 i2c_master_stop()之間,進行了類似隊列的保存操作(沒深入探究),然後是用i2c_master_cmd_begin()纔開始 進行硬件上的操作,所以在此處切換不同的IIC端口。一開始沒注意到此處,就想着前面怎麼沒有端口切換相關的參數呢。

剝離成中間函數

後面是用結構體來 模擬C++的類,進行函數封裝。
首先在 .h文件中定義結構體函數,並定義需要的結構體函數,然後是用 extern 做全局定義:


struct cenAngleGetBasicStruct{
	uint8_t AngleSenserNum;
	int8_t	SensorMoveDir[2];		//電機運動方向,+1:正向,-1:負向,0:不動
	float AngelSingleTurn[2];		//單圈記錄值,浮點數類型
	float AngelSingleTurnInit[2];	//單圈記錄值,精確到0.01度,放大100倍,
	long double AngelMultiTurn[2];		//多圈圈記錄角度值,精確到0.01度,放大100倍
	void (*cenAngleGetInit)(void);
	void (*cenAngleGet)(uint8_t sensorNum,float *returnAngle);

};

extern struct cenAngleGetBasicStruct cenAngleGetClass;  //全局使用的角度相關的結構體
void cenAngleGetClassInit(void);

然後在同名 .c 文件中進行賦值和函數綁定:

struct cenAngleGetBasicStruct cenAngleGetClass;  //全局使用的角度相關的結構體
/*------------------------------
*函數:cenAngleGetClassInit
*
*說明: 中間層的角度獲取class初始化
*		需要在主函數之前調用一次本函數,用於仿class類的初始化綁定
*
*
*參數:
*
*
*編寫:ZNZZ wcc 執念執戰 2023-5-20
*
*修改:
------------------------------*/
void cenAngleGetClassInit(void){
	//初始化內部參數,設置默認參數
	cenAngleGetClass.AngleSenserNum = 2;
	cenAngleGetClass.AngelMultiTurn[0]=0;

	//綁定對應的函數形成方法
	cenAngleGetClass.cenAngleGetInit = basAS5600Init;
	cenAngleGetClass.cenAngleGet = basAS5600AngleGet;  //角度獲取方法,在需要的地方調用即可
	//調用方法
	cenAngleGetClass.cenAngleGetInit();//調用Init方法來完成硬件初始化。
}


然後就能在其他地方包含 .h 頭文件後使用了。這種方法能夠較爲簡單的剝離底層和上層。實際上,將賦值這一步想辦法實現爲從bin地址中賦值爲指定的函數地址,理論上就能實現更爲複雜的類似驅動安裝的效果,實現跨平臺(當然,我沒有深究過別人的跨平臺是如何實現的,但是思路上應該差不多)。我後續的程序會盡可能的使用這種方式,將不同的功能都剝離成各自的仿類Class。這種方式寫起來略顯麻煩,因爲都要多加一步中間的結構體層。

下面是在mian中的測試方法:

void app_main(void)
{
	int count = 0;
	float angle = 0.0;
	cenAngleGetClassInit();
	while(1){
		cenAngleGetClass.cenAngleGet(count%2,&angle);//實現兩路IIC的快速切換讀取
		printf("GetAngle %d:%.2f \n",count%2,angle);
		count++;
	}
}

總結

我看到還有很多用C語言模擬C++或是其他語言特性的寫法,不過實現起來比較費時間,用起來也比較雞肋。所以簡單用一些比較好用的特性就夠了,畢竟現在很多單片機的編譯器都支持C++語法特性了,比如現在用的ESP32-IDF的開發環境就是支持C++語法的,只是需要簡單配置。

最近變得好懶了,FOC好幾周了,只是能讓電機保持不動了,怎麼讓電機動起來還是不想搞。

  • 本文水平有限,內容很多詞語由於知識水平問題不嚴謹或很離譜,但主要作爲記錄作用,希望以後的自己和路過的大神對必要的錯誤提出批評與指點,對可笑的錯誤請指出來,我會改正的。
  • 另外,轉載使用請註明作者和出處,不要刪除文檔中的關於作者的註釋。

隨夢,隨心,隨願,恆執念,爲夢執戰,執戰蒼天! ------------------執念執戰

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章