娇小w搡bbbb搡bbb,《第一次の人妻》,中国成熟妇女毛茸茸,边啃奶头边躁狠狠躁视频免费观看

歷史上的今天

今天是:2024年11月24日(星期日)

正在發生

2021年11月24日 | I2C協議->裸機程序->adapter驅動程序分析

發布者:電子設計藝術家 關鍵字:I2C協議  adapter 手機看文章 掃描二維碼
隨時隨地手機看文章

開發板:mini2440


內核  :linux2.6.32.2


參考  :韋東山畢業班I2C視頻教程


1、i2c協議簡要分析


    i2c中線是一種由 PHILIPS 公司開發的串行總線,用于連接微控制器及其外圍設備,它具有以下特點。

        1、只有兩條總線線路:一條串行數據線SDA,一條串行時鐘線SCL。

        2、每個連接到總線的器件都可以使用軟件根據它的唯一的地址來確定。

        3、傳輸數據的設備之間是簡單的主從關系。

        4、主機可以用作主機發送器或者主機接收器。

        5、它是一個真正的多主機總線,兩個或多個主機同時發起數據傳輸時,可以通過沖突檢測和仲裁來防止數據被破壞。

        6、串行的8位雙向傳輸,位速在標準模式下可達 100kbit/s,在快速模式下可達400kbit/s,在高速模式下可待3.4Mbit/s。

        7、片上的濾波器可以增加抗干擾能力,保證數據的完整性。

        8、連接到同一總線上的IC數量只受到總線的最大電容400Pf的限制。

如上圖所示,啟動一個傳輸時,主機先發送一個S信號,然后發送8位數據。這8位數據的前7位為從機地址,第八位表示傳輸的方向(0表示寫,1表示讀),如果有數據則繼續發送,最后發出P信號停止。

信號類型:


    注意:正常數據傳輸時,SDA 在 SCL 為低電平時改變,在 SCL 為高電平時保持穩定。

    開始信號 S 信號:

        SCL 為高電平時,SDA由高電平向低電平跳變,開始傳送數據。

    結束信號 P 信號:

        SCL 為高電平時,SDA由低電平向高電平跳變,結束傳送數據。

    響應信號 ACK:

        接收器在接收到8位數據后,在第9個時鐘周期,拉低 SDA 電平

    注意:在第9個時鐘周期,發送器保持SDA為高,如果有ACK,那么第9個時鐘周期SDA為低電平,如果沒有為高電平,發送器根據電平高低分辨是否有ACK信號。

          如果使能了IIC中斷,發送完8bit數據后,主機自動進入中斷處理函數,此時SCL被發送器拉低,讓接收器被迫等待。恢復傳輸只需要清除中斷掛起。


2、 s3c2440 讀寫流程

    1、設置傳輸模式 IICSTAT[7-6],我們做實驗與AT24C08通信時,2440作為主機,因此只用到主機發模式和主機收模式。

    2、寫入從機地址到 IICDS[7-1],此時IICDS[7-1]位表示從機地址,第0位不關心。如 AT24C08 為 0xA0(最低位寫0了,發送到數據線上的7位地址的后邊以為才表示收發,這里雖然寫0但并不是根據這里的0來真正發送的)。

    3、寫 0xF0(寫) 或 0xB0(讀)到 IICSTAT 寄存器, 高兩位表示 傳輸模式前邊設置過了,設置IICSTT[5-4] 為 11,使能傳輸,發送S信號。

    4、IIC控制器自動將第2步中設置的 IICDS[7-1] 再根據 傳輸模式 補充 IICDS[0]位,發送出去。

    5、進入第9個時鐘周期,此時,從機決定是否發出ACK信號,主機進入中斷,判斷是否收到ACK信號,以及是否繼續傳輸。

    繼續發送:

        1、將數據寫入 IICDS 

        2、清除中斷掛起,SCL時鐘恢復,IICDS的數據被自動發送到 SDA 線上,回到第5步。

    停止發送:

        1、寫 0xD0(寫) 和 0x90(讀) 到 IICATAT ,IICATAT[7-6]還是表示的傳輸模式,IICATAT[5-4] == 0 1,發送停止信號

        2、清除中斷掛起,SCL時鐘恢復,發出停止信號

        3、延時,等待停止信號發出

3、 AT24C08 讀寫分析


    1、寫過程

    寫過程與2440芯片的里的寫流程相一致,按照流程寫就OK


    2、讀過程

    讀過程是由2440芯片里的一個寫流程加一個讀流程組合而成,其中寫流程結束沒有發出P信號,而是直接發出了S信號開始讀流程,也就是我為什么加了一道紅線的原因。


附上一份簡單的裸機程序,僅供參考:基于MINI2440



#include

#include "s3c2440.h"

 

void Delay(int time);

 

#define WRDATA      (1)

#define RDDATA      (2)

 

typedef struct tI2C {

    unsigned char *pData;   /* 數據緩沖區 */

    volatile int DataCount; /* 等待傳輸的數據長度 */

    volatile int Status;    /* 狀態 */

    volatile int Mode;      /* 模式:讀/寫 */

    volatile int Pt;        /* pData中待傳輸數據的位置 */

}tS3C24xx_I2C, *ptS3C24xx_I2C;

 

static tS3C24xx_I2C g_tS3C24xx_I2C;

 

/*

 * I2C初始化

 */

void i2c_init(void)

{

    GPEUP  |= 0xc000;       // 禁止內部上拉

/*

 * AT24C08 兩根線 I2CSCL I2CSDA 與 2440芯片相連

 *  配置2440 GPECON GPE15 GPE14引腳為I2C功能

 */

    GPECON |= 0xa0000000;   // 選擇引腳功能:GPE15:IICSDA, GPE14:IICSCL

/* 開INT_IIC中斷 */

    //INTMSK &= ~(BIT_IIC);

 

    /* bit[7] = 1, 使能ACK

     * bit[6] = 0, IICCLK = PCLK/16

     * bit[5] = 1, 使能中斷

     * bit[3:0] = 0xf, Tx clock = IICCLK/16

     * PCLK = 50MHz, IICCLK = 3.125MHz, Tx Clock = 0.195MHz

     */

    IICCON = (1<<7) | (0<<6) | (1<<5) | (0xf);  // 0xaf

 

    //IICADD  = 0x10;     // S3C24xx slave address = [7:1]

    IICSTAT = 0x10;     // I2C串行輸出使能(Rx/Tx)

}

 

void I_Write(unsigned int slvaddr, unsigned char addr, unsigned char data)  

{  

    unsigned int ack;  

// 寫從地址

IICSTAT |= 0x1<<6;//主機寫模式   

    IICSTAT |= 0x1<<7;  

    IICDS = slvaddr;//0xa0;  //write slave address to IICDS   

    IICCON&=~0x10; //clear pending bit   

    IICSTAT = 0xf0;  //(M/T start)   

    while((IICCON & 1<<4) == 0);//udelay(10);//ack period and then interrupt is pending   

    // 寫寄存器地址    

    IICDS = addr;  

    IICCON&=~0x10; //clear pending bit   

    while((IICCON & 1<<4) == 0);//udelay(10);//ack period and then interrupt is pending   

// 寫數據

    IICDS = data;  

    IICCON&=~0x10; //clear pending bit   

    while((IICCON & 1<<4) == 0);//udelay(10);//ack period and then interrupt is pending   

// 發出停止信號

    IICSTAT = 0xD0; //write (M/T stop to IICSTAT)   

    IICCON&=~0x10; //clear pending bit   

 

    while((IICSTAT & 1<<5) == 1);  

}  

unsigned char I_Read(unsigned int slvaddr, unsigned char addr)  

{  

    unsigned char data  = 1;  

    int ack;  

// 寫從地址

IICSTAT |= 0x1<<6;//主機寫模式   

    IICSTAT |= 0x1<<7;  

slvaddr = 0xA0;    

    IICDS = slvaddr;//0xa0;  //write slave address to IICDS   

    IICCON&=~0x10; //clear pending bit   

    IICSTAT = 0xf0;  //(M/T start)   

    while((IICCON & 1<<4) == 0);//udelay(10);//ack period and then interrupt is pending   

  

    // 寫寄存器地址

    IICDS = addr;  

    IICCON&=~0x10; //clear pending bit   

    while((IICCON & 1<<4) == 0);//udelay(10);//ack period and then interrupt is pending   

 

    // 寫從地址(讀模式)

    slvaddr = 0xA1;

    IICSTAT &= ~(0x1<<6);//主機接受模式  

    IICSTAT |= 0x1<<7;  

    IICDS = slvaddr;  

    IICCON&=~0x10; //clear pending bit   

    IICSTAT = 0xb0;  //(M/R Start)   

    while((IICCON & 1<<4) == 0);//udelay(10);//uart_SendByte('o');//ack period and then interrupt is pending::   

 

// 讀數據

    data = IICDS;  

    //IICCON&=~0x10; //clear pending bit

IICCON = 0x2f; //清掛起狀態,并設置無應答

    while((IICCON & 1<<4) == 0);//udelay(10);//ack period and then interrupt is pending   

    data = IICDS;  

    

//IICCON&=~0x10; //clear pending bit   

IICCON = 0x2f; //清掛起狀態,并設置無應答

    while((IICCON & 1<<4) == 0);//udelay(10);//ack period and then interrupt is pending   

 

 

    IICSTAT = 0x90;  

IICCON = 0xaf;

    //IICCON &= ~0x10; //clear pending bit   

 

    while((IICSTAT & 1<<5) == 1);  

 

    return data;  

      



4、adapter驅動程序

    這里,我們主要分析驅動里的發送核心算法,至于注冊中斷,IO內存映射,設置寄存器不在討論。

    static int xxx_i2c_xfer(struct i2c_adapter *adpap, struct i2c_msg *msg,int num)

    這個算法函數的作用就是將上層封裝好的一些i2c_msg 進行解析,將數據寫入寄存器,發送出去。在設備驅動層,我們使用了類似i2c_smbus_write_byte 等函數,類似的函數有很多,它們的作用就是封裝i2c_msg 結構(比如讀和寫的 msg 肯定不一樣,讀一個字節和讀多個字節也不一樣),然后調用 i2c_smbus_xfer_emulated->i2c_transfer,最終調用到我們的xxx_i2c_xfer函數進行傳輸。通過分析i2c_smbus_xfer_emulated函數,我們可以了解i2c_msg是如何封裝的。下面,我們簡單分析一下,知道最上層想干什么,我們才能知道實現哪些底層的功能不是。

struct i2c_msg {

__u16 addr; //從機地址

__u16 flags;

__u16 len; // buf 里 有多少個字節

__u8 *buf; // 本 msg 含有的數據,可能是1個字節,可有可能是多個字節

};

    此函數,省略了很多內容,舉例分析而已~,細節請看源碼



static s32 i2c_smbus_xfer_emulated(struct i2c_adapter * adapter, u16 addr,

                                   unsigned short flags,

                                   char read_write, u8 command, int size,

                                   union i2c_smbus_data * data)

{

unsigned char msgbuf0[I2C_SMBUS_BLOCK_MAX+3];

unsigned char msgbuf1[I2C_SMBUS_BLOCK_MAX+2];

int num = read_write == I2C_SMBUS_READ?2:1; // 寫操作兩個Msg 讀操作一個msg 這和我們前面分析AT24c08是一致的

struct i2c_msg msg[2] = { { addr, flags, 1, msgbuf0 },

                          { addr, flags | I2C_M_RD, 0, msgbuf1 }

                        };

msgbuf0[0] = command; // 從機地址右移1位得到的,比如AT24C08  為 0x50

switch(size) {

case I2C_SMBUS_BYTE_DATA: // 單字節讀寫

if (read_write == I2C_SMBUS_READ)

msg[1].len = 1;

/*

* 讀:

* msgbuf0[0] = command

*  msg[1].len = 1 ,數據會讀到 msgbuf0[1] 里

*/

else {

msg[0].len = 2;

msgbuf0[1] = data->byte;

/*

* 寫:

* msgbuf0[0] = command

* msgbuf0[1] = data->byte

*/

}

break;

}

status = i2c_transfer(adapter, msg, num);

}

    上面代碼跟我們分析AT24C08的時候如出一轍,對于一個寫操作,我們只需要一個2440的寫流程對應于這里的一個Msg,然而對于讀操作,我們需要2440的兩個流程,對應于這里的兩個Msg。那么,我們底層控制器驅動需要做的工作就是,取出所有的Msg,將每一個Msg里buf里的數據發送出去,如果有下一個Msg, 那么再將下一個Msg里的buf發送完畢,最后發出P停止信號。還有一點,每發送一個Msg都要先發出S開始信號。


    在看adapter程序之前,我們先來簡單思考一下,發出S開始信號之后,可能有以下3中情況:

        1、當前msg.len == 0 ,如果有ACK直接發出stop信號。這種情況出現在,控制器枚舉設備的時候,因為它只發送S信號以及設備地址,不發送數據。

        2、根據msg->flags 為 I2C_M_RD 等信息判斷讀寫,msg->flags 最低位為1表示讀,最低位為0表示寫。

            #define I2C_M_TEN0x0010          /* this is a ten bit chip address */

            #define I2C_M_RD0x0001           /* read data, from slave to master */

[1] [2]
關鍵字:I2C協議  adapter 引用地址:I2C協議->裸機程序->adapter驅動程序分析

上一篇:i2c驅動程序全面分析,從adapter驅動程序到設備驅動程序
下一篇:Linux ARMv7中斷向量表搬移(2)

推薦閱讀

2018機器人創新生態年度峰會已于11月16日順利結束。本次峰會全天共組織六大主題演講,上午是“初心與展望”、“趨勢與探索”、“資本與機會”,下午為“生態與價值”、“合作與共融”、“創新與落地”。頭部企業、政府領導、知名學者以及優秀投資人分別從技術創新、產品應用、服務連接、資本對話等角度切入了機器人生態的細分領域,不僅為生態未來的發展貢...
(文章來源:中科羅伯特機器人學院) 優傲機器人公司在2005年創立于丹麥歐登塞,自從2008年發布首款協作式機器人(cobot)以來,方便易用的UR協作機器人的銷量呈現大幅增長,協作機器人將會是工業機器人發展中的翹楚。 汽車行業的自動化水平在制造業中一直是排在前列的。大多數汽車制造廠商在焊接、涂裝、沖壓等環節都會采用自動化技術。但隨著生產模式...
英飛凌科技股份公司(FSE:IFX / OTCQX:IFNNY)宣布推出業界滿足最新WPA3安全標準的專用Wi-Fi 4解決方案,擴展了其具有成本效益和出色性能的安全IoT解決方案產品組合。作為用于智能恒溫器或智能照明等物聯網應用的Wi-Fi /藍牙/ BLE組合芯片,全新英飛凌CYW43439解決方案將幫助產品設計師更好地遵守目前全球范圍內新涌現的安全法規,包括《加州消費者...
舒服的小睡,預示著一個令人不安的問題。在谷歌自動駕駛汽車項目進行的早期,該公司員工(你可以稱他們為beta測試人員)被允許開著配備自動駕駛技術的汽車回家。該項目的領導者們發現,同伴們如此迅速地過度信任了這項還處于萌芽階段的技術——他們選擇放松注意力,并盯著手機,而不是觀察前方的道路。“我們的發現相當可怕。”Waymo的前首席執行官約翰·...

史海拾趣

問答坊 | AI 解惑

AWR和CST將3D EM軟件集成到Microwave Office中

      Applied Wave Research Inc.是高頻電子設計自動化(EDA)工具提供商。該公司近期宣稱,它與位于德國達姆施塔特(Darmstadt)的Computer Simulation Technology (CST))合作,將三維(3D)電磁波(EM) 技術CST MIC ...…

查看全部問答∨

ADC0832

ADC0832_ADC0832…

查看全部問答∨

【EEWORLD模塊整理】+超聲波模塊

本帖最后由 paulhyde 于 2014-9-15 03:01 編輯 [local]1[/loc方便大家    …

查看全部問答∨

青越鋒與Altium在PCB方面研發能力比較

Altium為澳大利亞的一家為印刷電路板,提供輔助設計軟件的軟件生產商。旗下EDA產品Protel從1985年的單純的PCB設計的Dos版 Protel,到09年推出的包含PCB、Cam、FPGA等功能的Summer 09,中間經歷了多個版本的升級。就PCB方面的設計而言,Protel每一次 ...…

查看全部問答∨

故障問題

[原創]一個查找電路和器件故障機理的好儀器好方法 最近幫助一些客戶做電路故障的分析,在了解情況過程中,問起哪里壞了,大都能告訴我“某某器件壞了”,但再問“哪個管腳壞了”“可能是什么原因造成的”“壞到了啥程度”,往往就沒人接茬了,即 ...…

查看全部問答∨

請教關于電池的問題

大家有知道一次性電池中壽命最長的是那種嗎?使用壽命。最大能做到多少mAH呢? 先謝謝了 補充:不知道一次性電池的壽命是不是也是用mAh來衡量呢,如果不是,要怎么判斷呢? [ 本帖最后由 silencepiece 于 2010-6-4 16:52 編輯 ]…

查看全部問答∨

不到1500 中國版藍光碟機價格再跳水

本帖最后由 jameswangsynnex 于 2015-3-3 19:57 編輯 去年底曾創下藍光碟機價格新低的中國華錄集團,昨天又推出一款具有價格殺傷力的產品——售價1499元的藍光碟機。這不僅是目前中國市場上價格最低的藍光碟機,也使國內市場的藍光碟機價格首次降 ...…

查看全部問答∨

我不說不行啊---說吧

  姐妹們,小S代言的“睡睡瘦”也可以免費申領了!       姐妹們,我昨天又有一個大發現,“睡睡瘦”也可以免費申領了。   這可是個貨真價實的好東西,小S代言的,我以前買過,睡覺貼上就可以減肥,效果 ...…

查看全部問答∨

求助,80c51

用Proteus基于80c51弄一個8路搶答系統,第一個按按鈕的人對應的燈亮,其他人再按不會亮…

查看全部問答∨

求win ce 5.0下能用的usb接口藍牙模塊

如題 最好能詳細介紹一下 謝謝…

查看全部問答∨
小廣播
設計資源 培訓 開發板 精華推薦

最新單片機文章
何立民專欄 單片機及嵌入式寶典

北京航空航天大學教授,20余年來致力于單片機與嵌入式系統推廣工作。

 
EEWorld訂閱號

 
EEWorld服務號

 
汽車開發圈

 
機器人開發圈

電子工程世界版權所有 京ICP證060456號 京ICP備10001474號-1 電信業務審批[2006]字第258號函 京公網安備 11010802033920號 Copyright ? 2005-2025 EEWORLD.com.cn, Inc. All rights reserved
主站蜘蛛池模板: 瓦房店市| 鹤壁市| 湘潭市| 德昌县| 揭西县| 子长县| 达日县| 舒城县| 新宁县| 天柱县| 丰宁| 大名县| 枣强县| 乐亭县| 自贡市| 鹤山市| 平乐县| 嘉祥县| 黑龙江省| 铁岭县| 肇源县| 武邑县| 普兰县| 阳东县| 吉安县| 乌审旗| 恭城| 富民县| 新蔡县| 都昌县| 武邑县| 新昌县| 岳普湖县| 内江市| 穆棱市| 文山县| 扶余县| 额敏县| 丰都县| 嵊泗县| 香港|