欧美极品高清xxxxhd,国产日产欧美最新,无码AV国产东京热AV无码,国产精品人与动性XXX,国产传媒亚洲综合一区二区,四库影院永久国产精品,毛片免费免费高清视频,福利所导航夜趣136
標題:
關于stc32g12k128-beta單片機自帶can的ID過濾配置
[打印本頁]
作者:
內卷1234
時間:
2024-9-19 09:34
標題:
關于stc32g12k128-beta單片機自帶can的ID過濾配置
大家好,我最近剛接觸stc32g系列單片機,聽說這個單片機自帶兩個can口很方便,發的程序我試過了,通過外接一個收發器和官方給的例程可以精準的進行can的發送。但是我試了下濾單個id就是解決不了,請教一下大家該如何配置總線驗收寄存器。我需要過濾單個id為10040901報文
#include"STC32G.h"
#include "intrins.h"
typedef unsigned char u8;
typedef unsigned int u16;
typedef unsigned long u32;
#define MAIN_Fosc 24000000UL
/****************************** 用戶定義宏 ***********************************/
//CAN總線波特率=Fclk/((1+(TSG1+1)+(TSG2+1))*(BRP+1)*2)
#define TSG1 2 //0~15
#define TSG2 1 //1~7 (TSG2 不能設置為0)
#define BRP 3 //0~63
//24000000/((1+3+2)*4*2)=500KHz
#define SJW 1 //重新同步跳躍寬度
//總線波特率100KHz以上設置為 0; 100KHz以下設置為 1
#define SAM 0 //總線電平采樣次數: 0:采樣1次; 1:采樣3次
/*****************************************************************************/
/************* 本地常量聲明 **************/
#define Timer0_Reload (65536UL -(MAIN_Fosc / 1000)) //Timer 0 中斷頻率, 1000次/秒
#define STANDARD_FRAME 0 //幀格式:標準幀
#define EXTENDED_FRAME 1 //幀格式:擴展幀
/************* 本地變量聲明 **************/
typedef struct
{
u8 DLC:4; //數據長度, bit0~bit3
u8 :2; //空數據, bit4~bit5
u8 RTR:1; //幀類型, bit6
u8 FF:1; //幀格式, bit7
u32 ID; //CAN ID
u8 DataBuffer[8]; //數據緩存
}CAN_DataDef;
CAN_DataDef CAN1_Tx;
CAN_DataDef CAN1_Rx[8];
bit B_CanRead; //CAN 收到數據標志
bit B_1ms; //1ms標志
u16 msecond;
/************* 本地函數聲明 **************/
void CANInit();
u8 CanReadReg(u8 addr);
u8 CanReadMsg(CAN_DataDef *CAN);
void CanSendMsg(CAN_DataDef *CAN);
void Delay1000ms(void) //@24.000MHz
{
unsigned long edata i;
_nop_();
_nop_();
i = 5999998UL;
while (i) i--;
}
/********************* 主函數 *************************/
void main(void)
{
u8 sr,i,n;
WTST = 0; //設置程序指令延時參數,賦值為0可將CPU執行指令的速度設置為最快
EAXFR = 1; //擴展寄存器(XFR)訪問使能
CKCON = 0; //提高訪問XRAM速度
P0M1 = 0x30; P0M0 = 0x30; //設置P0.4、P0.5為漏極開路(實驗箱加了上拉電阻到3.3V)
P1M1 = 0x32; P1M0 = 0x32; //設置P1.1、P1.4、P1.5為漏極開路(實驗箱加了上拉電阻到3.3V), P1.1在PWM當DAC電路通過電阻串聯到P2.3
P2M1 = 0x3c; P2M0 = 0x3c; //設置P2.2~P2.5為漏極開路(實驗箱加了上拉電阻到3.3V)
P3M1 = 0x50; P3M0 = 0x50; //設置P3.4、P3.6為漏極開路(實驗箱加了上拉電阻到3.3V)
P4M1 = 0x3c; P4M0 = 0x3c; //設置P4.2~P4.5為漏極開路(實驗箱加了上拉電阻到3.3V)
P5M1 = 0x0c; P5M0 = 0x0c; //設置P5.2、P5.3為漏極開路(實驗箱加了上拉電阻到3.3V)
P6M1 = 0xff; P6M0 = 0xff; //設置為漏極開路(實驗箱加了上拉電阻到3.3V)
P7M1 = 0x00; P7M0 = 0x00; //設置為準雙向口
AUXR = 0x80; //Timer0 set as 1T, 16 bits timer auto-reload,
TH0 = (u8)(Timer0_Reload / 256);
TL0 = (u8)(Timer0_Reload % 256);
ET0 = 1; //Timer0 interrupt enable
TR0 = 1; //Tiner0 run
CANInit();
EA = 1; //打開總中斷
CAN1_Tx.FF = EXTENDED_FRAME; //擴展幀
CAN1_Tx.RTR = 0; //0:數據幀,1:遠程幀
CAN1_Tx.DLC = 0x08; //數據長度
CAN1_Tx.ID = 0x11040109; //CAN ID
CAN1_Tx.DataBuffer[0] = 0x00; //數據內容
CAN1_Tx.DataBuffer[1] = 0x00;
CAN1_Tx.DataBuffer[2] = 0x30;
CAN1_Tx.DataBuffer[3] = 0x00;
CAN1_Tx.DataBuffer[4] = 0x00;
CAN1_Tx.DataBuffer[5] = 0x00;
CAN1_Tx.DataBuffer[6] = 0x80;
CAN1_Tx.DataBuffer[7] = 0x00;
P54 = 0;
Delay1000ms();
P54 = 1;
while(1)
{
if(B_1ms) //1ms到
{
B_1ms = 0;
if(++msecond >= 1000) //1秒到
{
msecond = 0;
sr = CanReadReg(SR);
if(sr & 0x01) //判斷是否有 BS:BUS-OFF狀態
{
CANAR = MR;
CANDR &= ~0x04; //清除 Reset Mode, 從BUS-OFF狀態退出
}
else
{
CanSendMsg(&CAN1_Tx); //發送一幀數據
}
}
}
if(B_CanRead)
{
B_CanRead = 0;
n = CanReadMsg(CAN1_Rx); //讀取接收內容
if(n>0)
{
for(i=0;i<n;i++)
{
CanSendMsg(&CAN1_Rx[i]); //CAN總線原樣返回
}
}
}
}
}
/********************** Timer0 1ms中斷函數 ************************/
void timer0 (void) interrupt 1
{
B_1ms = 1; //1ms標志
}
//========================================================================
// 函數: u8 ReadReg(u8 addr)
// 描述: CAN功能寄存器讀取函數。
// 參數: CAN功能寄存器地址.
// 返回: CAN功能寄存器數據.
// 版本: VER1.0
// 日期: 2020-11-16
// 備注:
//========================================================================
u8 CanReadReg(u8 addr)
{
u8 dat;
CANAR = addr;
dat = CANDR;
return dat;
}
//========================================================================
// 函數: void WriteReg(u8 addr, u8 dat)
// 描述: CAN功能寄存器配置函數。
// 參數: CAN功能寄存器地址, CAN功能寄存器數據.
// 返回: none.
// 版本: VER1.0
// 日期: 2020-11-16
// 備注:
//========================================================================
void CanWriteReg(u8 addr, u8 dat)
{
CANAR = addr;
CANDR = dat;
}
//========================================================================
// 函數: void CanReadFifo(CAN_DataDef *CANx)
// 描述: 讀取CAN緩沖區數據函數。
// 參數: *CANx: 存放CAN總線讀取數據.
// 返回: none.
// 版本: VER2.0
// 日期: 2023-01-31
// 備注:
//========================================================================
void CanReadFifo(CAN_DataDef *CAN)
{
u8 i;
u8 pdat[5];
u8 RX_Index=0;
pdat[0] = CanReadReg((u8)(RX_BUF0 + (RX_Index++&3)));
if(pdat[0] & 0x80) //判斷是標準幀還是擴展幀
{
pdat[1] = CanReadReg((u8)(RX_BUF0 + (RX_Index++&3))); //擴展幀ID占4個字節
pdat[2] = CanReadReg((u8)(RX_BUF0 + (RX_Index++&3)));
pdat[3] = CanReadReg((u8)(RX_BUF0 + (RX_Index++&3)));
pdat[4] = CanReadReg((u8)(RX_BUF0 + (RX_Index++&3)));
CAN->ID = (((u32)pdat[1] << 24) + ((u32)pdat[2] << 16) + ((u32)pdat[3] << 8) + pdat[4]) >> 3;
}
else
{
pdat[1] = CanReadReg((u8)(RX_BUF0 + (RX_Index++&3))); //標準幀ID占2個字節
pdat[2] = CanReadReg((u8)(RX_BUF0 + (RX_Index++&3)));
CAN->ID = ((pdat[1] << 8) + pdat[2]) >> 5;
}
CAN->FF = pdat[0] >> 7; //幀格式
CAN->RTR = pdat[0] >> 6; //幀類型
CAN->DLC = pdat[0]; //數據長度
for(i=0;((i<CAN->DLC) && (i<8));i++) //讀取數據長度為len,最多不超過8
{
CAN->DataBuffer[i] = CanReadReg((u8)(RX_BUF0 + (RX_Index++&3))); //讀取有效數據
}
while(RX_Index&3) //判斷已讀數據長度是否4的整數倍
{
CanReadReg((u8)(RX_BUF0 + (RX_Index++&3))); //讀取填充數據,一幀數據占據4的整數倍緩沖區空間,不足補0
}
}
//========================================================================
// 函數: u8 CanReadMsg(void)
// 描述: CAN接收數據函數。
// 參數: *CANx: 存放CAN總線讀取數據..
// 返回: 幀個數.
// 版本: VER2.0
// 日期: 2023-01-31
// 備注:
//========================================================================
u8 CanReadMsg(CAN_DataDef *CAN)
{
u8 i;
u8 n=0;
do{
CanReadFifo(&CAN[n++]); //讀取接收緩沖區數據
i = CanReadReg(SR);
}while(i&0x80); //判斷接收緩沖區里是否還有數據,有的話繼續讀取
return n; //返回幀個數
}
//========================================================================
// 函數: void CanSendMsg(CAN_DataDef *CAN)
// 描述: CAN發送標準幀函數。
// 參數: *CANx: 存放CAN總線發送數據..
// 返回: none.
// 版本: VER1.0
// 日期: 2020-11-19
// 備注:
//========================================================================
void CanSendMsg(CAN_DataDef *CAN)
{
u32 CanID;
u8 RX_Index,i;
if(CAN->FF) //判斷是否擴展幀
{
CanID = CAN->ID << 3;
CanWriteReg(TX_BUF0,CAN->DLC|((u8)CAN->RTR<<6)|0x80); //bit7: 標準幀(0)/擴展幀(1), bit6: 數據幀(0)/遠程幀(1), bit3~bit0: 數據長度(DLC)
CanWriteReg(TX_BUF1,(u8)(CanID>>24));
CanWriteReg(TX_BUF2,(u8)(CanID>>16));
CanWriteReg(TX_BUF3,(u8)(CanID>>8));
CanWriteReg(TX_BUF0,(u8)CanID);
RX_Index = 1;
for(i=0;((i<CAN->DLC) && (i<8));i++) //數據長度為DLC,最多不超過8
{
CanWriteReg((u8)(TX_BUF0 + (RX_Index++&3)),CAN->DataBuffer[i]); //寫入有效數據
}
while(RX_Index&3) //判斷已讀數據長度是否4的整數倍
{
CanWriteReg((u8)(TX_BUF0 + (RX_Index++&3)),0x00); //寫入填充數據,一幀數據占據4的整數倍緩沖區空間,不足補0
}
}
else //發送標準幀
{
CanID = (u16)(CAN->ID << 5);
CanWriteReg(TX_BUF0,CAN->DLC|((u8)CAN->RTR<<6)); //bit7: 標準幀(0)/擴展幀(1), bit6: 數據幀(0)/遠程幀(1), bit3~bit0: 數據長度(DLC)
CanWriteReg(TX_BUF1,(u8)(CanID>>8));
CanWriteReg(TX_BUF2,(u8)CanID);
RX_Index = 3;
for(i=0;((i<CAN->DLC) && (i<8));i++) //數據長度為DLC,最多不超過8
{
CanWriteReg((u8)(TX_BUF0 + (RX_Index++&3)),CAN->DataBuffer[i]); //寫入有效數據
}
while(RX_Index&3) //判斷已讀數據長度是否4的整數倍
{
CanWriteReg((u8)(TX_BUF0 + (RX_Index++&3)),0x00); //寫入填充數據,一幀數據占據4的整數倍緩沖區空間,不足補0
}
}
CanWriteReg(CMR ,0x04); //發起一次幀傳輸
}
//========================================================================
// 函數: void CANSetBaudrate()
// 描述: CAN總線波特率設置函數。
// 參數: none.
// 返回: none.
// 版本: VER1.0
// 日期: 2020-11-19
// 備注:
//========================================================================
void CANSetBaudrate()
{
CanWriteReg(BTR0,(SJW << 6) + BRP);
CanWriteReg(BTR1,(SAM << 7) + (TSG2 << 4) + TSG1);
}
//========================================================================
// 函數: void CANInit()
// 描述: CAN初始化函數。
// 參數: none.
// 返回: none.
// 版本: VER1.0
// 日期: 2020-11-19
// 備注:
//========================================================================
void CANInit()
{
CANEN = 1; //CAN1模塊使能
CanWriteReg(MR ,0x05); //使能 Reset Mode, 采用單濾波設置
CANSetBaudrate(); //設置波特率
//設置過濾ID為:xxF8xxxx 的幀才接收
CanWriteReg(ACR0,0x80); //總線驗收代碼寄存器
CanWriteReg(ACR1,0x20);
CanWriteReg(ACR2,0x48);
CanWriteReg(ACR3,0x08);
CanWriteReg(AMR0,0x80); //總線驗收屏蔽寄存器
CanWriteReg(AMR1,0x20);
CanWriteReg(AMR2,0x48);
CanWriteReg(AMR3,0x08);
//取消過濾ID,所有幀都接收
// CanWriteReg(ACR0,0x00); //總線驗收代碼寄存器
// CanWriteReg(ACR1,0x00);
// CanWriteReg(ACR2,0x00);
// CanWriteReg(ACR3,0x00);
// CanWriteReg(AMR0,0xFF); //總線驗收屏蔽寄存器
// CanWriteReg(AMR1,0xFF);
// CanWriteReg(AMR2,0xFF);
// CanWriteReg(AMR3,0xFF);
CanWriteReg(IMR ,0xff); //中斷寄存器
CanWriteReg(ISR ,0xff); //清中斷標志
CanWriteReg(MR ,0x01); //退出 Reset Mode, 采用單濾波設置(設置過濾器后注意選擇濾波模式)
// CanWriteReg(MR ,0x00); //退出 Reset Mode, 采用雙濾波設置(設置過濾器后注意選擇濾波模式)
P_SW1 = (P_SW1 & ~(3<<4)) | (0<<4); //端口切換(CAN_Rx,CAN_Tx) 0x00:P0.0,P0.1 0x10:P5.0,P5.1 0x20:P4.2,P4.5 0x30:P7.0,P7.1
CANICR = 0x02; //CAN中斷使能
}
//========================================================================
// 函數: void CANBUS_Interrupt(void) interrupt CAN1_VECTOR
// 描述: CAN總線中斷函數。
// 參數: none.
// 返回: none.
// 版本: VER1.0
// 日期: 2020-11-19
// 備注:
//========================================================================
void CANBUS_Interrupt(void) interrupt CAN1_VECTOR
{
u8 isr;
u8 arTemp;
arTemp = CANAR; //CANAR現場保存,避免主循環里寫完 CANAR 后產生中斷,在中斷里修改了 CANAR 內容
isr = CanReadReg(ISR);
if((isr & 0x04) == 0x04) //TI
{
CANAR = ISR;
CANDR = 0x04; //CLR FLAG
}
if((isr & 0x08) == 0x08) //RI
{
CANAR = ISR;
CANDR = 0x08; //CLR FLAG
B_CanRead = 1;
}
if((isr & 0x40) == 0x40) //ALI
{
CANAR = ISR;
CANDR = 0x40; //CLR FLAG
}
if((isr & 0x20) == 0x20) //EWI
{
CANAR = ISR;
CANDR = 0x20; //CLR FLAG
}
if((isr & 0x10) == 0x10) //EPI
{
CANAR = ISR;
CANDR = 0x10; //CLR FLAG
}
if((isr & 0x02) == 0x02) //BEI
{
CANAR = ISR;
CANDR = 0x02; //CLR FLAG
}
if((isr & 0x01) == 0x01) //DOI
{
CANAR = ISR;
CANDR = 0x01; //CLR FLAG
}
CANAR = arTemp; //CANAR現場恢復
}
復制代碼
作者:
lrkeji1982
時間:
2024-9-19 11:57
手冊上有說明,這個是可以過濾的,我用過
作者:
內卷1234
時間:
2024-9-20 09:15
lrkeji1982 發表于 2024-9-19 11:57
手冊上有說明,這個是可以過濾的,我用過
精準的過濾一個很難算,我找到了一個軟件可以直接算出來,謝謝回復
歡迎光臨 (http://www.raoushi.com/bbs/)
Powered by Discuz! X3.1