SPI 讀取不同長度 寄存器_SPI協議,MCP2515裸機驅動詳解

be203e32-e60f-eb11-8da9-e4434bdf6706.png

SPI概述

Serial Peripheral interface 通用串行外圍設備接口

是Motorola首先在其MC68HCXX系列處理器上定義的。SPI接口主要應用在 EEPROM,FLASH,實時時鐘,AD轉換器,還有數字信號處理器和數字信號解碼器之間。

SPI,是一種高速的,全雙工,同步的通信總線,并且在芯片的管腳上只占用四根線,節約了芯片的管腳,同時為PCB的布局上節省空間。

SPI特點

采用主-從模式(Master-Slave) 的控制方式

SPI 規定了兩個 SPI 設備之間通信必須由主設備 (Master) 來控制次設備 (Slave). 一個 Master 設備可以通過提供 Clock 以及對 Slave 設備進行片選 (Slave Select) 來控制多個 Slave 設備, SPI 協議還規定 Slave 設備的 Clock 由 Master 設備通過 SCK 管腳提供給 Slave 設備, Slave 設備本身不能產生或控制 Clock, 沒有 Clock 則 Slave 設備不能正常工作。

而這里的SPI中的時鐘和相位,指的就是SCLk時鐘的特性,即保證主從設備兩者的時鐘的特性一致了,以保證兩者可以正常實現SPI通訊。

采用同步方式(Synchronous)傳輸數據

Master 設備會根據將要交換的數據來產生相應的時鐘脈沖(Clock Pulse), 時鐘脈沖組成了時鐘信號(Clock Signal) , 時鐘信號通過時鐘極性 (CPOL) 和 時鐘相位 (CPHA) 控制著兩個 SPI 設備間何時數據交換以及何時對接收到的數據進行采樣, 來保證數據在兩個設備之間是同步傳輸的.

c0203e32-e60f-eb11-8da9-e4434bdf6706.png

工作機制

概述

首先看下SPI Data Transfer模塊圖。

c3203e32-e60f-eb11-8da9-e4434bdf6706.png

上圖只是對 SPI 設備間通信的一個簡單的描述, 下面詳細解釋一下圖中所示的幾個組件(Module):

SSPBUF

Synchronous Serial Port Buffer, 泛指 SPI 設備里面的內部緩沖區, 一般在物理上是以 FIFO 的形式, 保存傳輸過程中的臨時數據;

c5203e32-e60f-eb11-8da9-e4434bdf6706.png

我們知道, 在每個時鐘周期內, Master 與 Slave 之間交換的數據其實都是 SPI 內部移位寄存器從 SSPBUF 里面拷貝的. 我們可以通過往 SSPBUF 對應的寄存器 (Tx-Data / Rx-Data register) 里讀寫數據, 間接地操控 SPI 設備內部的 SSPBUF。

例如, 在發送數據之前, 我們應該先往 Master 的 Tx-Data 寄存器寫入將要發送出去的數據, 這些數據會被 Master-SSPSR 移位寄存器根據 Bus-Width 自動移入 Master-SSPBUF 里, 然后這些數據又會被 Master-SSPSR 根據 Channel-Width 從 Master-SSPBUF 中移出, 通過 Master-SDO 管腳傳給 Slave-SDI 管腳, Slave-SSPSR 則把從 Slave-SDI 接收到的數據移入 Slave-SSPBUF 里. 與此同時, Slave-SSPBUF 里面的數據根據每次接收數據的大小(Channel-Width), 通過 Slave-SDO 發往 Master-SDI, Master-SSPSR 再把從 Master-SDI 接收的數據移入 Master-SSPBUF.在單次數據傳輸完成之后, 用戶程序可以通過從 Master 設備的 Rx-Data 寄存器讀取 Master 設備數據交換得到的數據。

SSPSR

Synchronous Serial Port Register, 泛指 SPI 設備里面的移位寄存器(Shift Regitser), 它的作用是根據設置好的數據位寬(bit-width) 把數據移入或者移出 SSPBUF;

c8203e32-e60f-eb11-8da9-e4434bdf6706.png

SSPSR 是 SPI 設備內部的移位寄存器(Shift Register). 它的主要作用是根據 SPI 時鐘信號狀態, 往 SSPBUF 里移入或者移出數據, 每次移動的數據大小由 Bus-Width 以及 Channel-Width 所決定。 Bus-Width 的作用是指定地址總線到 Master 設備之間數據傳輸的單位. 例如, 我們想要往 Master 設備里面的 SSPBUF 寫入 16 Byte 大小的數據: 首先, 給 Master 設備的配置寄存器設置 Bus-Width 為 Byte; 然后往 Master 設備的 Tx-Data 移位寄存器在地址總線的入口寫入數據, 每次寫入 1 Byte 大小的數據(使用 writeb 函數); 寫完 1 Byte 數據之后, Master 設備里面的 Tx-Data 移位寄存器會自動把從地址總線傳來的1 Byte 數據移入 SSPBUF 里; 上述動作一共需要重復執行 16 次。

Channel-Width 的作用是指定 Master 設備與 Slave 設備之間數據傳輸的單位. 與 Bus-Width 相似, Master 設備內部的移位寄存器會依據 Channel-Width 自動地把數據從 Master-SSPBUF 里通過 Master-SDO 管腳搬運到 Slave 設備里的 Slave-SDI 引腳, Slave-SSPSR 再把每次接收的數據移入 Slave-SSPBUF里.通常情況下, Bus-Width 總是會大于或等于 Channel-Width, 這樣能保證不會出現因 Master 與 Slave 之間數據交換的頻率比地址總線與 Master 之間的數據交換頻率要快, 導致 SSPBUF 里面存放的數據為無效數據這樣的情況

Controller

泛指 SPI 設備里面的控制寄存器, 可以通過配置它們來設置 SPI 總線的傳輸模式。 通常情況下, 我們只需要對上圖所描述的四個管腳(pin) 進行編程即可控制整個 SPI 設備之間的數據通信。

cc203e32-e60f-eb11-8da9-e4434bdf6706.png

Master 設備里面的 Controller 主要通過時鐘信號(Clock Signal)以及片選信號(Slave Select Signal)來控制 Slave 設備. Slave 設備會一直等待, 直到接收到 Master 設備發過來的片選信號, 然后根據時鐘信號來工作。

Master 設備的片選操作必須由程序所實現. 例如: 由程序把 SS/CS 管腳的時鐘信號拉低電平, 完成 SPI 設備數據通信的前期工作; 當程序想讓 SPI 設備結束數據通信時, 再把 SS/CS 管腳上的時鐘信號拉高電平.

SCK

Serial Clock, 主要的作用是 Master 設備往 Slave 設備傳輸時鐘信號, 控制數據交換的時機以及速率;

SS/CS

Slave Select/Chip Select, 用于 Master 設備片選 Slave 設備, 使被選中的 Slave 設備能夠被 Master 設備所訪問。

SDO/MOSI

Serial Data Output/Master Out Slave In, 在 Master 上面也被稱為 Tx-Channel, 作為數據的出口, 主要用于 SPI 設備發送數據。

SDI/MISO

Serial Data Input/Master In Slave Out, 在 Master 上面也被稱為 Rx-Channel, 作為數據的入口, 主要用于SPI 設備接收數據。

SPI 設備在進行通信的過程中, Master 設備和 Slave 設備之間會產生一個數據鏈路回環(Data Loop), 就像上圖所畫的那樣, 通過 SDO 和 SDI 管腳, SSPSR 控制數據移入移出 SSPBUF, Controller 確定 SPI 總線的通信模式, SCK 傳輸時鐘信號。

極性和相位

要想搞清楚SPI的數據傳輸,首先要搞清楚相位和極性的概念,即SPI的極性Polarity和相位Phase。

最常見的寫法是CPOL和CPHA,不過也有一些其他寫法,簡單總結如下:

  • (1) CKPOL (Clock Polarity) = CPOL = POL = Polarity = (時鐘)極性
  • (2) CKPHA (Clock Phase) = CPHA = PHA = Phase = (時鐘)相位
  • (3) SCK=SCLK=SPI的時鐘
  • (4) Edge=邊沿,即時鐘電平變化的時刻,即上升沿(rising edge)或者下降沿(falling edge)

對于一個時鐘周期內,有兩個edge,分別稱為:

  • (1)Leading edge=前一個邊沿=第一個邊沿,對于開始電壓是1, 那么就是1變成0的時候,對于開始電壓是0,那么就是0變成1的時候;
  • (2)Trailing edge=后一個邊沿=第二個邊沿,對于開始電壓是1, 那么就是0變成1的時候(即在第一次1變成0之后,才可能有后面的0變成1), 對于開始電壓是0,那么就是1變成0的時候;

本博文采用如下用法:

  1. 極性=CPOL
  2. 相位=CPHA
  3. SCLK=時鐘
  4. 第一個邊沿和第二個邊沿

CPOL和CPHA,分別都可以是0或時1,對應的四種組合就是:

ble data-draft-node="block" data-draft-type="table" data-size="normal" data-row-style="normal">

d0203e32-e60f-eb11-8da9-e4434bdf6706.png

CPOL=0,CPHA=0

脈沖傳輸前和完成后都保持在低電平狀態,所以 CPOL=0,即低電平是空閑時的電平。在第一個邊沿(上升沿)采樣數據,第二個邊沿(下降沿)輸出數據,對應著CPHA=0。

d3203e32-e60f-eb11-8da9-e4434bdf6706.png

CPOL=0,CPHA=1

脈沖傳輸前和完成后都保持在低電平狀態,所以 CPOL=0,即低電平是空閑時的電平。在第二個邊沿(下降沿)采樣數據,第一個邊沿(上升沿)輸出數據,對應著CPHA=1。

d4203e32-e60f-eb11-8da9-e4434bdf6706.png

CPOL=1,CPHA=0

脈沖傳輸前和完成后都保持在高電平狀態,所以 CPOL=1,即高電平是空閑時的電平。在第一個邊沿(下降沿)采樣數據,第二個邊沿(上升沿)輸出數據,對應著CPHA=0。

d6203e32-e60f-eb11-8da9-e4434bdf6706.png

CPOL=1,CPHA=1

脈沖傳輸前和完成后都保持在高電平狀態,所以 CPOL=1,即高電平是空閑時的電平。在第二個邊沿(上升沿)采樣數據,第一個邊沿(下降沿)輸出數據,對應著CPHA=1。

da203e32-e60f-eb11-8da9-e4434bdf6706.png

軟件中如何設置SPI的極性和相位

SPI分主設備和從設備,兩者通過SPI協議通訊。 而設置SPI的模式,是從設備的模式,決定了主設備的模式。 所以要先去搞懂從設備的SPI是何種模式,然后再將主設備的SPI的模式,設置和從設備相同的模式,即可正常通訊。

對于從設備的SPI是什么模式,有兩種:

  • (1)固定的,有SPI從設備硬件決定的 SPI從設備,具體是什么模式,相關的datasheet中會有描述,需要自己去datasheet中找到相關的描述,即: 關于SPI從設備,在空閑的時候,是高電平還是低電平,即決定了CPOL是0還是1; 然后再找到關于設備是在上升沿還是下降沿去采樣數據,這樣就是,在定了CPOL的值的前提下,對應著可以推算出CPHA是0還是1了。
  • (2)可配置的,由軟件自己設定 從設備也是一個SPI控制器,4種模式都支持,此時只要自己設置為某種模式即可。 然后知道了從設備的模式后,再去將SPI主設備的模式,設置為和從設備模式一樣即可。 對于如何配置SPI的CPOL和CPHA的話,不多細說,多數都是直接去寫對應的SPI控制器中對應寄存器中的CPOL和CPHA那兩位,寫0或寫1即可。

數據交換(Data Exchanges)

SPI只有主模式和從模式之分,沒有讀和寫的說法,因為實質上每次SPI是主從設備在交換數據。也就是說,你發一個數據必然會收到一個數據;你要收一個數據必須也要先發一個數據。

SPI 設備間的數據傳輸之所以又被稱為數據交換, 是因為 SPI 協議規定一個 SPI 設備不能在數據通信過程中僅僅只充當一個 "發送者(Transmitter)" 或者 "接收者(Receiver)"。 在每個 Clock 周期內, SPI 設備都會發送并接收一個 bit 大小的數據, 相當于該設備有一個 bit 大小的數據被交換了。

一個 Slave 設備要想能夠接收到 Master 發過來的控制信號, 必須在此之前能夠被 Master 設備進行訪問 (Access)。 所以, Master 設備必須首先通過 SS/CS pin 對 Slave 設備進行片選, 把想要訪問的 Slave 設備選上。 在數據傳輸的過程中, 每次接收到的數據必須在下一次數據傳輸之前被采樣. 如果之前接收到的數據沒有被讀取, 那么這些已經接收完成的數據將有可能會被丟棄, 導致 SPI 物理模塊最終失效。

因此, 在程序中一般都會在 SPI 傳輸完數據后, 去讀取 SPI 設備里的數據, 即使這些數據(Dummy Data)在我們的程序里是無用的。

SPI舉例

下面舉一個例子幫助大家理解。

SPI是一個環形總線結構,由ss(cs)、sck、sdi、sdo構成,其時序其實很簡單,主要是在sck的控制下,兩個雙向移位寄存器進行數據交換。

假設下面的8位寄存器裝的是待發送的數據10101010,上升沿發送、下降沿接收、高位先發送。 那么第一個上升沿來的時候 數據將會是sdo=1;寄存器=0101010x。下降沿到來的時候,sdi上的電平將所存到寄存器中去,那么這時寄存器=0101010sdi,這樣在 8個時鐘脈沖以后,兩個寄存器的內容互相交換一次。這樣就完成里一個spi時序。

舉例: 假設主機和從機初始化就緒:并且主機的sbuff=0xaa,從機的sbuff=0x55,下面將分步對spi的8個時鐘周期的數據情況演示一遍:假設上升沿發送數據

dd203e32-e60f-eb11-8da9-e4434bdf6706.png

這樣就完成了兩個寄存器8位的交換,上面的上表示上升沿、下表示下降沿,sdi、sdo相對于主機而言的。

下一步就是把 上面的過程轉為動畫

e0203e32-e60f-eb11-8da9-e4434bdf6706.png

e3203e32-e60f-eb11-8da9-e4434bdf6706.png

e6203e32-e60f-eb11-8da9-e4434bdf6706.png

eb203e32-e60f-eb11-8da9-e4434bdf6706.png

請仔細比較下交換后的bit順序。

了解了SPI協議說明之后,下面我們基于Cortex-A9架構的exynos-4412,講解SPI控制器的使用。

Cortex-A9 SPI控制器

硬件設計

本例是基于FS4412開發板,SPI控制器外接了MCP2515,MCP2515與exynos-4412的硬件連接圖如下圖所示。

ed203e32-e60f-eb11-8da9-e4434bdf6706.png

ef203e32-e60f-eb11-8da9-e4434bdf6706.png

f3203e32-e60f-eb11-8da9-e4434bdf6706.png

管腳連接說明

由上圖可知SPI個引腳與SOC的pin連接關系:

  • CS <---------> BUF_BK_LED <---------> GPC1_2
  • SO <---------> BUF_I2C_SDA6 <---------> GPC1_3
  • SI <---------> BUF_I2C_SCL6 <---------> GPC1_4
  • SCK <---------> BUF_GPC1_1 <---------> GPC1_1
  • INT <-------------------------------------------> BUF_GPX0_0
  • MCP2515芯片連接在4412芯片的SPI2上。
  • 中斷連接在GPX0_0上; CS、SO、SI、SCK復用了GPIO引腳GPC1的引腳。

MCP2515輸出連接SN65HVD230 CAN總線收發器,SN65HVD230是德州儀器公司生產的3.3V CAN收發器。為了節省功耗,縮小電路體積,MCP2515 CAN總線控制器的邏輯電平采用LVTTL,SN65HVD230就是與其配套的收發器。

Cortex-A9 SPI控制器

exynos4412 scp中的串行外設接口(SPI)通過各種外設來傳輸串行數據。SPI包括兩個8、16和32位移位寄存器,用于傳輸和接收數據。在SPI傳輸過程中,它同時傳輸(串行移出)和接收(串行移位)數據。

特性

  • 全雙工
  • 用于Tx/Rx的8/16/32位移位寄存器
  • 支持8位/16位/32位總線接口
  • 支持摩托羅拉SPI協議和National Semiconductor Microwire
  • 兩個獨立的32位寬的發送和接收FIFO:端口0的深度為64,端口1和2中的深度為16
  • 主模式和從模式
  • 接收而不發送操作
  • 發送/接收最高頻率為50 MHz

SPI的操作

SPI在Exynos 4412 SCP和外部設備之間傳輸1位串行數據。 Exynos 4412 SCP中的SPI支持CPU或DMA分別同時發送或接收FIFO和雙向傳輸數據。 SPI有兩個信道,即Tx信道和Rx信道。Tx信道有來自Tx的路徑

FIFO到外部設備。Rx通道有從外部設備到Rx FIFO的路徑。

CPU(或DMA)必須將數據寫入寄存器SPI_TX_DATA,才能將數據寫入FIFO。寄存器上的數據會自動移動到Tx FIFO。要從Rx FIFO讀取數據,CPU(或DMA)必須訪問寄存器SPI_RX_DATA,數據會自動發送到SPI_RX_DATA寄存器。

CMU寄存器可以控制SPI的工作頻率。

操作模式

SPI有兩種模式,即主模式和從模式。 在主模式下,生成SPICLK并將其傳輸到外部設備。XspiCS#是選擇從機的信號,它指示在設置XspiCS時數據有效低水平。在發送或接收數據包之前,必須將XspiCS設置為低。

FIFO存取

SPI支持對fifo的CPU訪問和DMA訪問。 對fifo的CPU訪問和DMA訪問的數據大小從8位、16位或32位數據中選擇。當它選擇8位數據大小時,有效位是0到7比特。用戶可以定義觸發閾值來引發CPU中斷。端口0中每個FIFO的觸發電平由從0到252字節的步進為4個字節,端口1中每個FIFO的字節從0到63字節按1個字節的步長設置。 SPI_MODE_CFG寄存器的TxDMAOn或RxDMAOn位必須設置為使用DMA訪問。DMA訪問支持只有單次傳輸和4突發傳輸。在Tx FIFO中,DMA請求信號是高的,直到Tx FIFO滿為止。在Rx FIFO中,如果FIFO不為空,DMA請求信號高。

片選控制

芯片選擇XspiCS是激活的低信號。換句話說,當XspiCS輸入為0時,選擇芯片。 您可以自動或手動控制XspiCS。不需要改變。 手動模式 當您使用手動控制模式時,您應清除AUTO_N_MANUAL(默認值為0)。NSSOUT位控制XspiCS級別。 自動模式 使用自動控制模式時,必須將AUTO_N_MANUAL設置為1。XspiCS在數據包和自動打包。NCS_TIME_COUNT 控制XspiCS的非激活期。NSSOUT在此時不可用。

SPI寄存器說明

配置寄存器CH_CFGn

f4203e32-e60f-eb11-8da9-e4434bdf6706.png

【bit:5】 軟件復位的時候必須先將此位設置為1,給一個延時后再設置為0; 【bit:4】 設置SPI端口是主機還是從機,0:主機,1:從機; 【bit:2-3】設置相位和極性; 【bit:1】接收數據的通道使能位; 【bit:0】發送數據的通道使能位。

數據寬度寄存器MODE_CFGn

設置總線數據寬度,一般設置為1個字節。

f7203e32-e60f-eb11-8da9-e4434bdf6706.png

片選寄存器CS_REGn

只有NSSOUT 拉低,才會進行數據的傳輸

f8203e32-e60f-eb11-8da9-e4434bdf6706.png

移位寄存器狀態寄存器SPI_STATUSn

發送操作開始,如果移位寄存器空了,該值置1,通過該值判斷數據是否發送出去。

fa203e32-e60f-eb11-8da9-e4434bdf6706.png

發送緩沖寄存器SPI_TX_DATA

fb203e32-e60f-eb11-8da9-e4434bdf6706.png

接收緩沖寄存器SPI_RX_DATA

fd203e32-e60f-eb11-8da9-e4434bdf6706.png

SPI初始化流程

  1. 設置GPIO引腳為SPI模式;
  2. 設置clock;
  3. 軟件復位;
  4. 設置CPOL CPHA 為 00 模式,并設置為主機模式;
  5. 設置數據位;
  6. 片選。

1. 設置GPIO引腳為SPI模式

因為CS、SO、SI、SCK復用了GPIO引腳GPC1的引腳。首先需要設置GPIO引腳為SPI模式。

01213e32-e60f-eb11-8da9-e4434bdf6706.png

如上所示,該寄存器設置方式如下:

GPC1.CON = (GPC1.CON & ~0xffff0) | 0x55550;//設置IO引腳為SPI模式

Step 1 設置CPOL CPHA 為 00 模式

08213e32-e60f-eb11-8da9-e4434bdf6706.png

SPI2.CH_CFG&=~((0x1<< 4)|(0x1<<3)|(0x1<< 2)|0x3);

//master mode, CPOL = 0, CPHA = 0 (Format A)

2 設置clock

時鐘的設置需要依賴鎖相環(PLL)時鐘產生器。

0b213e32-e60f-eb11-8da9-e4434bdf6706.png

從30.2.1節可知,時鐘配置需要參考CMU這一章。

10213e32-e60f-eb11-8da9-e4434bdf6706.png

14213e32-e60f-eb11-8da9-e4434bdf6706.png

由上圖可知SPI 的clock源是SCLK_SPI。

搜索SCLK_SPI

17213e32-e60f-eb11-8da9-e4434bdf6706.png

從第七章搜所有的SPI

18213e32-e60f-eb11-8da9-e4434bdf6706.png

繼續搜索

69213e32-e60f-eb11-8da9-e4434bdf6706.png

由此可知SPI0~2的之中受CLKMPLL_USER_T 控制,繼續搜索。 而此時鐘位于寄存器CLK_SRC_PERIL1的bit[27:24]。

6c213e32-e60f-eb11-8da9-e4434bdf6706.png

繼續搜索SPI2。

76213e32-e60f-eb11-8da9-e4434bdf6706.png

79213e32-e60f-eb11-8da9-e4434bdf6706.png

搜索到CLK_SRC_MASK_PERIL1、CLK_DIV_PERIL2。 可知寄存器CLK_SRC_MASK_PERIL1默認是打開的,所以不用設置。 CLK_DIV_PERIL2用來設置SPI2分頻的,于是寄存器設置如下:

CLK_SRC_PERIL1 = (CLK_SRC_PERIL1 & ~(0xF<<24)) | 6<<24;
// 0x6: 0110 = SCLKMPLL_USER_T 800Mhz
CLK_DIV_PERIL2 = 19 <<8 | 3;//SPI_CLK = 800/(19+1)/(3+1)

3. 軟件復位

7d213e32-e60f-eb11-8da9-e4434bdf6706.png
void soft_reset(void)
{ SPI2.CH_CFG |= 0x1 << 5;delay(1);                     //延時SPI2.CH_CFG &= ~(0x1 << 5);
}

4. 設置CPOL CPHA 為 00 模式,并設置為主機模式

SPI2.CH_CFG &= ~( (0x1 << 4) | (0x1 << 3) | (0x1 << 2) | 0x3);

5. 設置數據位MODE_CFG

82213e32-e60f-eb11-8da9-e4434bdf6706.png
SPI2.MODE_CFG &= ~((0x3 << 17) | (0x3 << 29));//BUS_WIDTH=8bit,CH_WIDTH=8bit 

6. 片選

SPI2.CS_REG &= ~(0x1 << 1);        //選擇手動選擇芯片

初始化搞定后,下面我們來看如何利用SPI收發數據。

收發數據流程

收發數據流程如下:

  1. spi復位
  2. 片選從機
  3. 收發數據
  4. 取消片選

【注意】一下代碼為設置每次只讀寫1個byte數據,每次讀寫1個byte數據都要按照上述步驟操作。

1 spi復位

void soft_reset(void)
{ SPI2.CH_CFG |= 0x1 << 5;delay(1);                     //延時SPI2.CH_CFG &= ~(0x1 << 5);
}

2 片選從機

void slave_enable(void)
{SPI2.CS_REG &= ~0x1; //enable salvedelay(3);
}

3.1 發送數據

void send_byte(unsigned char data)
{SPI2.CH_CFG |= 0x1; // enable Tx Channeldelay(1);SPI2.SPI_TX_DATA = data;while( !(SPI2.SPI_STATUS & (0x1 << 25)) );SPI2.CH_CFG &= ~0x1; // disable Tx Channel
}

3.2 接收數據

unsigned char recv_byte()
{unsigned char data;SPI2.CH_CFG |= 0x1 << 1; // enable Rx Channeldelay(1);data = SPI2.SPI_RX_DATA;delay(1);SPI2.CH_CFG &= ~(0x1 << 1); //disable Rx Channelreturn  data;
}

取消片選

void slave_disable(void)
{SPI2.CS_REG |= 0x1; //disable salvedelay(1);
}

OK,到底為止,如何通過SPI收發數據,我們已經可以實現了,但是SPI往往外部回接各種各樣的從設備。下面我們來看SPI如何操作從設備。

上面我們說了,fs4412開發板的SPI2外設外接了MCP2515,現在我們來分析一下MCP2515。

MCP2515

MCP2515詳細資料,大家自行搜索 MCP2515 datasheet 《帶有 SPI 接口的獨立 CAN 控制器》。

簡介

MCP2515是一種獨立的CAN總線通信控制器,是Microchip公司首批獨立CAN解決方案的升級器件,其傳輸能力較Microchip公司原有CAN控制器(MCP2510)高兩倍,高通信速率可達到1Mbps。MCP2515能夠接收和發送標準數據幀和擴展數據幀以及遠程幀,通過兩個接收屏蔽寄存器和六個接收過濾寄存器濾除無關報文,從而減輕CPU負擔。

特性

MCP2515主要功能參數及電氣特性如下:

  • (1)支持CAN技術規范2.0A/B, 高傳輸速率達到1Mbps;
  • (2)支持標準數據幀、擴展數據幀和遠程幀,每幀數據域長度可為0~8個字節;
  • (3)內含兩個的接收緩沖器和三個發送緩沖器,并且可編程設定優先級;
  • (4)內含六個29位(bit)的接收過濾寄存器和兩個29位(bit)的接收屏蔽寄存器;
  • (5)高速SPI接口,支持SPI 0,0和1,1模式;
  • (6)一次性模式可確保報文被一次性傳輸;
  • (7)具有可編程時鐘脈沖輸出引腳,可作為其他芯片時鐘信號源;
  • (8) 幀起始(SOF)信號輸出功能可被用于在確定的系統中(如時間觸發CAN-TTCAN)執行時隙功能,或在CAN總線診斷中決定早期總線出級;
  • (9) 采用低功耗CMOS技術,工作電壓:2.7V~5.5V, 工作電流:5mA(待機狀態1μA);
  • (10)工作溫度范圍:(I)-40℃到+85℃,(E)-40℃到+125℃。

結構框圖

88213e32-e60f-eb11-8da9-e4434bdf6706.png

CAN模塊

MCP2515 是一款獨立 CAN 控制器, 可簡化需要與 CAN總線連接的應用。如上圖 簡要顯示了 MCP2515 的結構框圖。該器件主要由三個部分組成:

  1. CAN 模塊,包括 CAN 協議引擎、驗收濾波寄存 器、驗收屏蔽寄存器、發送和接收緩沖器。
  2. 用于配置該器件及其運行的控制邏輯和寄存器。
  3. SPI 協議模塊。

CAN 模塊的功能是處理所有 CAN 總線上的報文接收和發送。報文發送時,首先將報文裝載到正確的報文緩沖器和控制寄存器中。通過 SPI 接口設置控制寄存器中的相應位或使用發送使能引腳均可啟動發送操作。通過讀取相應的寄存器可以檢查通訊狀態和錯誤。 會對在 CAN總線上檢測到的任何報文進行錯誤檢查,然后與用戶定義的濾波器進行匹配,以確定是否將報文移到兩個接收緩沖器中的一個。

SPI 協議模塊

MCU通過SPI接口與該器件連接。使用標準的SPI讀/寫指令以及專門的 SPI 命令來讀 / 寫所有的寄存器。 MCP2515 設計可與許多單片機的串行外設接口( SPI)直接相連,支持 0,0 和 1,1 運行模式。外部數據和命令通過 SI 引腳傳送到器件中,且數據在 SCK 時鐘信號的上升沿傳送進去。 MCP2515 在 SCK 的下降沿通過 SO引腳傳送出去。在進行任何操作時, CS 引腳都必須保持為低電平。

與MCP2515通信

SPI指令集

我們要想操作MCP2515,只能用過SPI總線向MCP2515發送一些數據,根據規定,發送不同的數據就代表不同的操作,于是就形成了對應的指令集。

89213e32-e60f-eb11-8da9-e4434bdf6706.png

如上圖所示,比如我們要執行復位操作,那么我只需要通過SPI總線寫入11000000,即0xC0即可。

下面我們來詳細分析如何向MCP2515發送復位命令,如何讀寫數據。

復位

因為復位只需要發送0XC0即可,并不需要其他額外操作,所以我們只需要根據我們之前章節所講的SPI發送數據流程操作接口。

void reset_2515()
{soft_reset();      //復位spi控制器slave_enable() ;   //片選從機send_byte(0xc0);   //發送復位命令slave_disable() ;  //取消片選}

讀取數據

8d213e32-e60f-eb11-8da9-e4434bdf6706.png

由上圖可知,讀取數據流程如下:

  1. 片選從機
  2. 通過SPI發送指令0x03
  3. 發送地址
  4. 讀取數據
  5. 取消片選
unsigned char read_byte_2515(unsigned char Addr)
{unsigned char ret;slave_enable();send_byte(0x03);send_byte(Addr);ret = recv_byte();slave_disable();return(ret);
}

發送數據

90213e32-e60f-eb11-8da9-e4434bdf6706.png

由上圖可知,發送數據流程如下:

  1. 片選從機
  2. 通過SPI發送指令0x02
  3. 發送地址
  4. 發送數據
  5. 取消片選
void write_byte_2515(unsigned char addr,unsigned char data)
{slave_enable();send_byte(0x02);send_byte(addr);send_byte(data);slave_disable();
}

讀RX緩沖器

93213e32-e60f-eb11-8da9-e4434bdf6706.png

裝載TX緩沖器

96213e32-e60f-eb11-8da9-e4434bdf6706.png

請求發送(RTS)指令

9b213e32-e60f-eb11-8da9-e4434bdf6706.png

位修改指令

a6213e32-e60f-eb11-8da9-e4434bdf6706.png

有些寄存器要修改對應的bit,必須先發送該指令,發送屏蔽字節,然后才可以修改屏蔽字節中對應位為1的值,后面會有例子給出。

CAN

知道該如何和CAN通信,下面我們來看如何初始化CAN。

CAN初始化

CAN的初始化步驟如下:

  1. MCP2515復位
  2. 設置MCP2515為配置模式
  3. 位定時配置,有配置寄存器CNF1,CNF2,CNF3控制
  4. 中斷使能
  5. 接收緩沖器配置
  6. 引腳控制寄存器和狀態寄存器

此外為方便測試,我們再加一步: 7. 回環模式,用于測試

1. MCP2515復位

上一節已經實現。

void reset_2515()

2. 設置MCP2515為配置模式

ab213e32-e60f-eb11-8da9-e4434bdf6706.png

由上圖所示,只需要向地址0X0F寫入數據0x80即可。

 write_byte_2515(0x0f, 0x80);    //CANCTRL寄存器--進入配置模式 中文DATASHEET 58頁

3. 位定時配置,有配置寄存器CNF1,CNF2,CNF3控制

CAN總線上的所有節點都必須具有相同的標稱比特率。CAN 協議采用不歸零( Non Return to Zero, NRZ)編碼方式,在數據流中不對時鐘信號進行編碼。因此,接收時鐘信號必須由接收節點恢復并與發送器的時鐘同步。

由于不同節點的振蕩器頻率和傳輸時間不同,接收器應具有某種能與數據傳輸邊沿同步的鎖相環( Phase Lock Loop, PLL)來同步時鐘并保持這種同步。

鑒于數據采用 NRZ 編碼,有必要進行位填充以確保至少每 6 位時間發生一次邊沿,使數字鎖相環 ( Digital Phase LockLoop, DPLL)同步。MCP2515 通過 DPLL 實現位定時。 DPLL 被配置成同輸入數據同步,并為發送數據提供標稱定時。 DPLL 將每一 個 位 時 間 分 割 為 由 最 小 單 位 為 時 間 份 額 ( Time Quanta, TQ)所組成的多個時間段。

在位時間幀中執行的總線定時功能,例如與本地振蕩器同步、網絡傳輸延遲補償和采樣點定位等,都是由DPLL 的可編程位定時邏輯來規定的。

CNF1 BRP<5:0> 控制波特率預分頻比的設置。這些位根據OSC1 輸入頻率設置 TQ 的時間長度。當 BRP<5:0> =‘b000000’, TQ 最小值取 2 TOSC。通過 SJW<1:0> 選擇以 TQ 計的同步跳轉寬度。

CNF2 PRSEG<2:0> 位設定以 TQ 計的傳播段時間長度。PHSEG1<2:0>位設定以TQ計的相位緩沖段PS1的時間長度。

CNF3 如果 CNF2.BTLMODE 位為 1,則相位緩沖段 PS2 的時間長度將由 PHSEG2<2:0> 位設定,以 TQ 計。如果BTLMODE 位為 0,則 PHSEG2<2:0> 位不起作用。

b0213e32-e60f-eb11-8da9-e4434bdf6706.png

b7213e32-e60f-eb11-8da9-e4434bdf6706.png

bd213e32-e60f-eb11-8da9-e4434bdf6706.png

MCP2515 波特率配置以16M晶振為例, SJW 段數(1+PRSEG+PRSEG1+PRSEG2)

#define CNF1_20K     0xd3        //4   20(1+4+8+7) 
#define CNF2_20K     0xfb          
#define CNF3_20K     0x46           //可以設置的波特率 5K 10K 15K 20K 25K 40K 50K 80K 100K 125K 200K 400K 500K 667K 800K 1M
write_byte_2515(0x2A, CNF1_20K); //CNF1位定時配置寄器  
write_byte_2515(0x29, CNF2_20K); //CNF2位定時配置寄器 
write_byte_2515(0x28, CNF3_20K); //CNF3位定時配置寄器 

4. 中斷使能

MCP2515 有八個中斷源。 CANINTE 寄存器包含了使能各中斷源的中斷使能位。 CANINTF 寄存器包含了各中斷源的中斷標志位。當發生中斷時, INT 引腳將被MCP2515 拉為低電平, 并保持低電平狀態直至 MCU 清除中斷。中斷只有在引起相應中斷的條件消失后,才會被清除。 建議在對 CANINTF 寄存器中的標志位進行復位操作時,采用位修改命令而不要使用一般的寫操作。這是為了避免在寫命令執行過程中無意間修改了標志位,進而導致中斷丟失。 應該注意的是, CANINTF 中的中斷標志位是可讀寫位,因此在相關 CANINTE 中斷使能位置 1 的前提下,對上述任一位置 1 均可使 MCU 產生中斷請求。

bf213e32-e60f-eb11-8da9-e4434bdf6706.png
write_byte_2515(0x2B, 0x1f);     //CANINTE中斷使能寄存器  

5. 接收緩沖器配置

MCP2515 具有兩個全接收緩沖器。每個接收緩沖器配備 有多 個 驗 收濾 波 器。除 上述 專 用 接收 緩 沖 器外,MCP2515 還具有單獨的報文集成緩沖器 ( Message Assembly Buffer, MAB) ,可作為第三個接收緩沖器。

c3213e32-e60f-eb11-8da9-e4434bdf6706.png

bit5:6設置為1,其余位暫時不用,設置為0.

 write_byte_2515(0x60, 0x60);   //RXB0CTRL接收緩沖器0 控制寄存器 

6. 引腳控制寄存器和狀態寄存器

當引腳配置為數字輸出引腳時,相應的接收緩沖器中的BFPCTRL.BxBFM位應被清零, 而BFPCTRL.BnBFE位應被置 1。在這種工作模式下,引腳的狀態由 BFPCTRL.BnBFS位控制。 BnBFS位寫入1時,將使相應的緩沖器滿中斷引腳輸出高電平,寫入 0 將使該引腳輸出低電平。當引腳處于這種模式時,該引腳的狀態只應通過位修改 SPI 命令來修改,以避免任何緩沖器滿中斷引腳出現干擾。

c8213e32-e60f-eb11-8da9-e4434bdf6706.png
void bit_modify_2515(unsigned char addr, unsigned char mask, unsigned char data)
{
//    CS_SPI = 0 ;slave_enable() ;send_byte(0x05) ;send_byte(addr) ;send_byte(mask) ;send_byte(data) ;slave_disable() ;
//    CS_SPI = 1 ;
}
bit_modify_2515(0x0C, 0x0f, 0x0f); //BFPCTRL_RXnBF 引腳控制寄存器和狀態寄存器 中文DATASHEET 29 頁

7. 回環模式,用于測試

write_byte_2515(0x0f, 0x40);   //CAN控制寄存器--回環模式,用于測試

can緩沖區數據收發

MCP2515 采用三個發送緩沖器。每個發送緩沖器占用14 字節的 SRAM,并映射到器件存儲器中。 其中第一個字節 TXBnCTRL 是與報文緩沖器相關的控制寄存器。該寄存器中的信息決定了報文在何種條件下發送,并在報文發送時指示其狀態 。

用 5 個字節來裝載標準和擴展標識符以及其他報文仲裁信息(見寄存器 3-3 到寄存器 3-7) 。最后 8 個字節用于裝載等待發送報文的 8 個可能的數據字節 。

至少須將 TXBnSIDH、 TXBnSIDL 和 TXBnDLC 寄存器裝載數據。如果報文包含數據字節,還需要對 TXBnDm寄存器進行裝載。若報文采用擴展標識符,應對 TXBnEIDm 寄存器進行裝載,并將 TXBnSIDL.EXIDE 位置1。

下面我們看如何向CAN的緩沖區0發送和接收數據。

數據發送

can緩沖區0數據發送流程如下:

  1. 設置為發送最高優先級
  2. 設置發送緩沖器0標準標識符高位
  3. 設置發送緩沖器0標準標識符低位
  4. 設置發送緩沖器0數據長度碼8字節
  5. 向緩沖區寫入數據,地址從0x36起
  6. 發送請求命令0x81,發送數據

1. 設置為發送最高優先級

ca213e32-e60f-eb11-8da9-e4434bdf6706.png
write_byte_2515(0x30, 0x03); //設置為發送最高優先級

2. 設置發送緩沖器0標準標識符高位

cf213e32-e60f-eb11-8da9-e4434bdf6706.png
write_byte_2515(0x31, 0xff); //發送緩沖器0標準標識符高位

3. 設置發送緩沖器0標準標識符低位

d2213e32-e60f-eb11-8da9-e4434bdf6706.png
write_byte_2515(0x32, 0x00); //發送緩沖器0標準標識符低位

4. 設置發送緩沖器0數據長度碼8字節

d6213e32-e60f-eb11-8da9-e4434bdf6706.png
write_byte_2515(0x35, 0x08);  //發送緩沖器0數據長度碼8字節

5. 向緩沖區寫入數據,地址從0x36起

d8213e32-e60f-eb11-8da9-e4434bdf6706.png
write_byte_2515(0x36+i ,tx_buff[i]); //向txb緩沖器中寫入8個字節

6. 發送請求命令0x81,發送數據

void send_req_2515()
{//   CS_SPI = 0; //復位soft_reset();      //復位spi控制器slave_enable() ;   //片選從機send_byte(0x81);   //發送請求命令slave_disable() ;  //取消片選
// CS_SPI=1;
}

Can數據的接收

從CAN緩沖區讀取數據流程如下:

  1. 讀取中斷標志寄存器0x2c的value,判斷bit0是否為1
  2. 從接收緩沖區讀走數據,地址從0X66開始
  3. 軟復位
  4. 向中斷標志寄存器0x2c寫入位掩碼
  5. 向中斷標志寄存器0x2c,寫入數據0,清終端

1. 讀取中斷標志寄存器0x2c的value,判斷bit0是否為1

當報文傳送至某一接收緩沖器時,與該接收緩沖器對應的 CANINTF.RXnIF 位將置 1。一旦緩沖器中的報文處理完畢, MCU 就必須將該位清零以接收下一條報文。該控制位提供的鎖定功能確保 MCU 尚未處理完上一條報文前, MCP2515 不會將新的報文載入接收緩沖器。

如果 CANINTE.RXnIE 位被置 1,器件會在 INT 引腳產生一個中斷,顯示接收到報文有效。另外,如果被配置為接收緩沖器滿中斷引腳,與之相應的 RXnBF 引腳會被拉低。

MCP2515 有八個中斷源。 CANINTE 寄存器包含了使能各中斷源的中斷使能位。 CANINTF 寄存器包含了各中斷源的中斷標志位。當發生中斷時, INT 引腳將被MCP2515 拉為低電平, 并保持低電平狀態直至 MCU 清除中斷。中斷只有在引起相應中斷的條件消失后,才會被清除。

建議在對 CANINTF 寄存器中的標志位進行復位操作時,采用位修改命令而不要使用一般的寫操作。這是為了避免在寫命令執行過程中無意間修改了標志位,進而導致中斷丟失。

應該注意的是, CANINTF 中的中斷標志位是可讀寫位,因此在相關 CANINTE 中斷使能位置 1 的前提下,對上述任一位置 1 均可使 MCU 產生中斷請求。

da213e32-e60f-eb11-8da9-e4434bdf6706.png

2. 從接收緩沖區讀走數據

rx_buff[i]= read_byte_2515(0x66+i);

3. 軟復位

soft_reset();

4. 向中斷標志寄存器0x2c寫入位掩碼

bit_modify_2515(0x2c,0x01,0x00);//修改bit 0

5.清中斷

write_byte_2515(0x2c, 0x00);

最終操作代碼如下

節省篇幅,重復函數不貼了。

#define CNF1_20K     0xd3        //4   20(1+4+8+7) 
#define CNF2_20K     0xfb          
#define CNF3_20K     0x46  void  Init_can(void)
{reset_2515(); //復位write_byte_2515(0x0f, 0x80); //CANCTRL寄存器--進入配置模式 中文DATASHEET 58頁//可以設置的波特率 5K 10K 15K 20K 25K 40K 50K 80K 100K 125K 200K 400K 500K 667K 800K 1Mwrite_byte_2515(0x2A, CNF1_20K); //CNF1位定時配置寄器   中文DATASHEET 41-42頁write_byte_2515(0x29, CNF2_20K); //CNF2位定時配置寄器   中文DATASHEET 41-42頁write_byte_2515(0x28, CNF3_20K); //CNF3位定時配置寄器   中文DATASHEET 41-43頁write_byte_2515(0x2B, 0x1f);     //CANINTE中斷使能寄存器  中文DATASHEET 50 頁write_byte_2515(0x60, 0x60);     //RXB0CTRL接收緩沖器0 控制寄存器 中文DATASHEET 27 頁//write_byte_2515(0x70, 0x20);   //接收緩沖器1控制寄存器bit_modify_2515(0x0C, 0x0f, 0x0f); //BFPCTRL_RXnBF 引腳控制寄存器和狀態寄存器 中文DATASHEET 29 頁write_byte_2515(0x0f, 0x40);   //CAN控制寄存器--回環模式,用于測試
}
void send_byte(unsigned char data)
{SPI2.CH_CFG |= 0x1; // enable Tx Channeldelay(1);SPI2.SPI_TX_DATA = data;while( !(SPI2.SPI_STATUS & (0x1 << 25)) );SPI2.CH_CFG &= ~0x1; // disable Tx Channel
}
unsigned char recv_byte()
{unsigned char data;SPI2.CH_CFG |= 0x1 << 1; // enable Rx Channeldelay(1);data = SPI2.SPI_RX_DATA;delay(1);SPI2.CH_CFG &= ~(0x1 << 1); //disable Rx Channelreturn  data;
}
void bit_modify_2515(unsigned char addr, unsigned char mask, unsigned char data)
{
//    CS_SPI = 0 ;slave_enable() ;send_byte(0x05) ;send_byte(addr) ;send_byte(mask) ;send_byte(data) ;slave_disable() ;
//    CS_SPI = 1 ;
}
void Can_send(unsigned char *tx_buff)
{unsigned char i;write_byte_2515(0x30, 0x03); //設置為發送最高優先級write_byte_2515(0x31, 0xff); //發送緩沖器0標準標識符高位write_byte_2515(0x32, 0x00); //發送緩沖器0標準標識符低位write_byte_2515(0x35, 0x08);  //發送緩沖器0數據長度碼8字節for(i = 0; i < 8; i++){write_byte_2515(0x36+i ,tx_buff[i]); //向txb緩沖器中寫入8個字節
//  printf("%x ",tx_buff[i]);}send_req_2515();}unsigned char Can_receive(unsigned char *rx_buff)
{unsigned char i,flag;flag = read_byte_2515(0x2c); //CANINTF——中斷標志寄存器printf("flag=%xn",flag);//  printf(" SPI2.SPI_STATUS =%xn", SPI2.SPI_STATUS );//   soft_reset();if (flag&0x1)                //接收緩沖器0滿中斷標志位{for(i = 0; i < 16; i++){rx_buff[i]= read_byte_2515(0x66+i);//  printf("%x ",rx_buff[i]);//  printf(" SPI2.SPI_STATUS =%xn", SPI2.SPI_STATUS );soft_reset();}bit_modify_2515(0x2c,0x01,0x00);write_byte_2515(0x2c, 0x00);if (!(rx_buff[1]&0x08)) return(1);   //接收標準數據幀}return(0);
}
int main(void)
{GPX2.CON = 0x1 << 28;uart_init();unsigned char ID[4],buff[8];                 //狀態字unsigned char key;unsigned char ret;//,j,k,ret0,ret1,ret2,ret3,ret4,ret5,ret6,ret7,ret8,ret9;unsigned int rx_counter;volatile int i=0;GPC1.CON = (GPC1.CON & ~0xffff0) | 0x55550;//設置IO引腳為SPI模式/*spi clock config*/CLK_SRC_PERIL1 = (CLK_SRC_PERIL1 & ~(0xF<<24)) | 6<<24;// 0x6: 0110 = SCLKMPLL_USER_T 800MhzCLK_DIV_PERIL2 = 19 <<8 | 3;//SPI_CLK = 800/(19+1)/(3+1)soft_reset();                    // 軟復位SPI控制器SPI2.CH_CFG &= ~( (0x1 << 4) | (0x1 << 3) | (0x1 << 2) | 0x3);//master mode, CPOL = 0, CPHA = 0 (Format A)SPI2.MODE_CFG &= ~((0x3 << 17) | (0x3 << 29));   //BUS_WIDTH=8bit,CH_WIDTH=8bitSPI2.CS_REG &= ~(0x1 << 1);        //選擇手動選擇芯片mydelay_ms(10);    //延時Init_can();   //初始化MCP2515printf("n************ SPI CAN test!! ************n");while(1){//Turn on ledGPX2.DAT = GPX2.DAT | 0x1 << 7;mydelay_ms(50);printf("nplease input 8 bytesn");for(i=0;i<8;i++){src[i] = getchar();putc(src[i]);}printf("n");Can_send(src); //發送標準幀mydelay_ms(100);ret = Can_receive(dst); //接收CAN總線數據printf("ret=%xn",ret);printf("src=");for(i=0;i<8;i++) printf(" %x", src[i]);//將CAN總線上收到的數據發到串行口printf("n");printf("dst=");for(i=0;i<8;i++) printf(" %x",dst[6+i]); //將CAN總線上收到的數據發到串行口printf("n");//Turn offGPX2.DAT = GPX2.DAT & ~(0x1 << 7);mydelay_ms(100);} //while(1)return 0;
} //main

問題

1 什么是feddback 時鐘?

2 Can的緩沖寄存器組 原理 更多信息,請關注 公號 一口Linux

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/454715.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/454715.shtml
英文地址,請注明出處:http://en.pswp.cn/news/454715.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

oracle并發執行max,跪求大量并發執行insert into select語句的方案

現在有數十萬張表要從A庫通過insert into tablename select * from tablenamedblink的方式導入到B庫中。B機上80個cpu&#xff0c;160G內存。希望能夠大量并發執行。怎么寫腳本呢&#xff1f;誰有這方面的經驗&#xff0c;麻煩指點一下。謝謝。下面是我的腳本&#xff1a;#!/us…

20162314 《Program Design Data Structures》Learning Summary Of The First Week

20162314 2017-2018-1 《Program Design & Data Structures》Learning Summary Of The First Week Summary of teaching materials Algorithm analysis is the basic project of the computer science.Increasing function prove that the utilization of the time and spa…

高并發解決方法

2019獨角獸企業重金招聘Python工程師標準>>> 高并發來說&#xff0c;要從實際項目的每一個過程去考慮&#xff0c;頁面&#xff0c;訪問過程&#xff0c;服務器處理&#xff0c;數據庫訪問每個過程都可以處理。&#xff08;前端-寬帶-后端-DB&#xff09; 集群&…

MySQL 之 存儲過程

一、初識存儲過程 1、什么是存儲過程 存儲過程是在大型數據庫系統中一組為了完成特定功能的SQL語句集&#xff0c;存儲在數據庫中。存儲過程經過第一次編譯后&#xff0c;再次調用不需要編譯&#xff0c;用戶可以通過指定的存儲過程名和給出一些存儲過程定義的參數來使用它。…

C/C++面試感受和經驗以及面試題收藏

http://topic.csdn.net/u/20080924/15/3b00a84e-970f-4dea-92f2-868c5d1ad825.html 前段時間剛參加了n多公司的C/C軟件工程師的面試&#xff0c;有國企&#xff0c;外企&#xff0c;私企&#xff08;moto&#xff0c;飛思卡爾&#xff0c;港灣&#xff0c;中國衛星XXX&#xf…

oracle存儲過程季度方法,Oracle存儲過程、觸發器實現獲取時間段內周、月、季度的具體時間...

歡迎技術交流。 QQ&#xff1a;138986722創建table&#xff1a;create table tbmeetmgrinfo(id number primary key, /*主鍵&#xff0c;自動增加 */huiyishi number, /*會議室編號 */STARTTIME varchar2(30), /*會議開始時間 */ENDTIME varchar2(30), /*會議結束時間 */CREATE…

如何root安卓手機_安卓Root+卡開機畫面救磚教程丨以一加手機為例

一加手機買到手已經用了1個多月了&#xff0c;還有很多朋友在問我怎么Root、怎么替換Recovery、怎么安裝Magisk、有時候刷Magisk模塊變磚怎么解救。小編統一整理一下&#xff0c;其他安卓手機也可以參考&#xff0c;很多思路都是通用的。一加手機刷入TWRP并RootTWRP大概是現在安…

Linux用ctrl + r 查找以前(歷史)輸入的命令

在Linux系統下一直用上下鍵查找以前輸入的命令&#xff0c;這個找剛輸入不久的命令還是很方便的&#xff0c;但是比較久遠的命令&#xff0c;用上下鍵效率就不高了。那個history命令也是個花架子&#xff0c;雖然功能多&#xff0c;但不好用&#xff0c;網上找了下&#xff0c;…

敬畏生產環境

生產環境即是運行用戶正在使用的系統環境。有時&#xff0c;一些粗心的用戶會把他們的生產環境交給我們&#xff0c;希望我們來進行變更操作&#xff0c;如部署Agent等。但我們實際上并不清楚這些生產環境有多重要&#xff0c;有時也不清楚我們的變更操作有多危險&#xff0c;其…

unity消息隊列判斷字符串相等有錯誤_Python3十大經典錯誤及解決辦法

◆ ◆ ◆ ◆ ◆接觸了很多Python愛好者&#xff0c;有初學者&#xff0c;亦有轉行人。不論大家學習Python的目的是什么&#xff0c;總之&#xff0c;學習Python前期寫出來的代碼不報錯就是極好的。下面&#xff0c;嚴小樣兒為大家羅列出Python3十大經典錯誤及解決辦法&#xf…

php qmqp 沒有方法,CentOS7 php 安裝 amqp擴展

繼續安裝完 rabbitmq后&#xff0c;安裝phpqmqp擴展1.安裝rabbitmq-c安裝最新版wget -c https://github.com/alanxz/rabbitmq-c/releases/download/v0.8.0/rabbitmq-c-0.8.0.tar.gztar zxf rabbitmq-c-0.8.0.tar.gzcd rabbitmq-c-0.8.0./configure --prefix/usr/local/rabbitmq…

如何提高UDP的可靠性

TCP是通過確認機制和超時重傳機制實現可靠傳輸 UDP UDP它不屬于連接型協議&#xff0c;因而具有資源消耗小&#xff0c;處理速度快的優點&#xff0c;所以通常音頻、視頻和普通數據在傳送時使用UDP較多&#xff0c;因為它們即使偶爾丟失一兩個數據包&#xff0c;也不會對接收結…

CentOS 7鏡像下載

方式一 官網下載 官網鏈接&#xff1a;http://isoredirect.centos.org/centos/7/isos/x86_64/ Actual Country 國內資源 Nearby Countries 周邊國家資源 方式二 阿里云下載 阿里云站點&#xff1a;http://mirrors.aliyun.com/centos/7/isos/x86_64/ 各個版本的ISO鏡像文件…

Docker Dirty Cow逃逸

2019獨角獸企業重金招聘Python工程師標準>>> 在Linux中&#xff0c;有一個功能&#xff1a;VDSO(virtual dvnamic shared object),這是一個小型共享庫&#xff0c;能將內核自動映射到所有用戶程序的地址空間。 Docker逃逸利用Dirty Cow漏洞&#xff0c;將Payload寫到…

九月騰訊,創新工場,淘寶等公司最新面試三十題(更新至10.04)

九月騰訊&#xff0c;創新工場&#xff0c;淘寶等公司最新面試三十題 引言 曾記否&#xff0c;去年的10月份也同此刻一樣&#xff0c;是找工作的高峰期&#xff0c;本博客便是最初由整理微軟等公司面試題而發展而來的。如今&#xff0c;又即將邁入求職高峰期--10月份&#…

oracle 存long,ORACLE中LONG類型字段的存取

&#xfeff;&#xfeff;Oracle中存取4000字節以上大文本類型可以用此數據類型&#xff0c;其在C#中的讀寫方法如下&#xff1a;注意需要引用 System.Data.OracleClient然后添加命名空間&#xff1a;using System.Data.OracleClientORALCE建庫腳本&#xff1a;CREATE TABLE TE…

創建office一直轉圈_Windows寫字板出現廣告條幅:推薦用戶使用在線版Office

自Windows 95開始&#xff0c;寫字板(Wordpad)應用就一直預裝在Windows操作系統中。它是一款非常簡單的文本編輯器&#xff0c;在功能方面介于記事本和Word之間。近日Rafael Rivera發現微軟正在為這款古老的寫字板添加新功能--在應用中添加廣告橫幅。這個廣告橫幅就是推薦那些寫…

2017軟件工程實踐第二次作業

1、 項目地址&#xff1a;https://github.com/one-piece-zero/sudoku 2、PSP表格記錄的估計耗時 3、解題思路&#xff1a; 在拿到這個題目的時候&#xff0c;我最早想到的是大一下學期做的程序語言綜合設計實踐中的N皇后問題&#xff0c;這兩個題目之間有許多的類似之處&#x…

CentOS7 安裝或遷移 wordpress(完整遷移)

一、安裝Apache web服務器 安裝Apache web服務器&#xff1a; yum install -y httpd # 使用yum安裝 systemctl start httpd # 啟動Apache服務器 systemctl enable httpd # Apache服務器開機后自動啟動 使用瀏覽器打開http://127.0.0.1檢查Apache安裝是否成功。成功后…

WinForm部署問題

WinForm部署問題 1、解決&#xff1a;This implementation is not part of the Windows Platform 問題&#xff1f; 一&#xff1a;單擊 開始 &#xff0c;單擊 運行 &#xff0c;鍵入 gpedit.msc &#xff0c;然后單擊 確定 。    二&#xff1a;依次展開 計算機配置 &…