📄 canbus.cpp
字号:
/*
** INTEL CONFIDENTIAL
** Copyright 2000-2003 Intel Corporation All Rights Reserved.
**
** The source code contained or described herein and all documents
** related to the source code (Material) are owned by Intel Corporation
** or its suppliers or licensors. Title to the Material remains with
** Intel Corporation or its suppliers and licensors. The Material contains
** trade secrets and proprietary and confidential information of Intel
** or its suppliers and licensors. The Material is protected by worldwide
** copyright and trade secret laws and treaty provisions. No part of the
** Material may be used, copied, reproduced, modified, published, uploaded,
** posted, transmitted, distributed, or disclosed in any way without Intel抯
** prior express written permission.
** No license under any patent, copyright, trade secret or other intellectual
** property right is granted to or conferred upon you by disclosure or
** delivery of the Materials, either expressly, by implication, inducement,
** estoppel or otherwise. Any license under such intellectual property rights
** must be express and approved by Intel in writing.
*/
#include <windows.h>
#include <types.h>
#include <drv_glob.h>
#include <nkintr.h>
#include <oalintr.h>
#include <MCP2510.h>
unsigned short ReadBuf[256] ;
TCHAR read_buf[256] ;
//liudiping
static void Delay(USHORT count)
{
volatile int i, j = 0;
volatile static int loop = 400000000/100000;
for(;count > 0;count--)
for(i=0;i < loop; i++) { j++; }
}
/****************************************************************************
【功能说明】SPI接口IO片选初始化
****************************************************************************/
void MCP2510_IO_CS_Init( void )
{
MCP2510_CS_OUT ;
MCP2510_SI_OUT ;
MCP2510_SCK_OUT ;
MCP2510_SO_IN ;
MCP2510_SO_PULLUP ; //允许上拉
//MCP2510_SO_DISPULLUP ; //禁止上拉
MCP2510_SI_L ; //SI put 0
MCP2510_SCK_L ; //SCK put 0
{ U16 k=0; for( ; k <= DELAY_TIME; k++ ) ; } //延时至少300ns
MCP2510_CS_H ; // unselect the MCP2510
{ U16 k=0; for( ; k <= DELAY_TIME; k++ ) ; } //延时至少300ns
}
/****************************************************************************
【功能说明】SPI接口读写开始,片选有效
****************************************************************************/
void MCP2510_RW_Start( void )
{
MCP2510_SI_L ; //SI put 0
MCP2510_SCK_L ; //SCK put 0
{ U16 k=0; for( ; k <= DELAY_TIME; k++ ) ; } //延时至少300ns
MCP2510_CS_L ; // Select the MCP2510
{ U16 k=0; for( ; k <= DELAY_TIME; k++ ) ; } //延时至少300ns
}
/****************************************************************************
【功能说明】SPI接口写入数据
****************************************************************************/
void Spi_Write( U8 Data )
{
U8 m ;
for( m = 0; m < 8; m++ )
{
if( (Data&0x80)==0x80 )
MCP2510_SI_H; //SI put 1
else
MCP2510_SI_L; //SI put 0
{ U16 k=0; for( ; k <= DELAY_TIME; k++ ) ; } //延时至少300ns
MCP2510_SCK_H ; //SCK put 1
Data = Data<<1 ;
MCP2510_SCK_L ; //SCK put 0
{ U16 k=0; for( ; k <= DELAY_TIME; k++ ) ; } //延时至少300ns
}
}
/****************************************************************************
【功能说明】SPI接口读出数据
****************************************************************************/
U8 Spi_Read( )
{
U8 m ;
U8 data = 0 ;
for( m = 0; m < 8; m++ )
{
MCP2510_SCK_H ; //SCK put 1
{ U16 k=0; for( ; k <= DELAY_TIME; k++ ) ; } //延时至少300ns
data = data<<1;
if( MCP2510_SO_GET != 0 )
data |= 0x01 ;
else
data &= 0xfe;
{ U16 k=0; for( ; k <= DELAY_TIME; k++ ) ; } //延时至少300ns
MCP2510_SCK_L ; //SCK put 0
{ U16 k=0; for( ; k <= DELAY_TIME; k++ ) ; } //延时至少300ns
}
return (data);
}
/****************************************************************************
【功能说明】 Send Command to MCP2510 via SPI
****************************************************************************/
void SendCMDMCP2510( U8 CMD )
{
MCP2510_RW_Start() ; //Initial IO port and CS is select
Spi_Write( CMD );
MCP2510_CS_H ; // Deselect the MCP2510
}
/****************************************************************************
【功能说明】软件复位MCP2510
****************************************************************************/
void MCP2510_Reset()
{
MCP2510_RW_Start() ;
Spi_Write( MCP2510INSTR_RESET );
MCP2510_CS_H ;
}
/****************************************************************************
【功能说明】向MCP2510指定地址写入一个字节
****************************************************************************/
void MCP2510_Write( U8 address, U8 value)
{
MCP2510_RW_Start() ;
Spi_Write(MCP2510INSTR_WRITE);
Spi_Write( address );
Spi_Write( value );
MCP2510_CS_H ;
}
/****************************************************************************
【功能说明】修改指定地址寄存器的某些位
****************************************************************************/
void MCP2510_WriteBits( U8 address, U8 data, U8 mask )
{
MCP2510_RW_Start() ;
Spi_Write( MCP2510INSTR_BITMDFY );
Spi_Write( address);
Spi_Write( mask);
Spi_Write( data);
MCP2510_CS_H ;
}
/****************************************************************************
【功能说明】 Read often used status
//Status 7 6 5 4 3 2 1 0
// | | | | | | | |
// | | | | | | | |___CANINTF.RX0IF
// | | | | | | |_______CANINTF.RX1IF
// | | | | | |___________TXB0CTRL.TXREQ
// | | | | |_______________CANINTF.TX0IF
// | | | |___________________TXB1CTRL.TXREQ
// | | |_______________________CANINTF.TX1IF
// | |___________________________TXB2CTRL.TXREQ
// |_______________________________CANINTF.TX2IF
****************************************************************************/
unsigned char MCP2510_ReadStatus()
{
unsigned char result;
MCP2510_RW_Start() ;
Spi_Write(MCP2510INSTR_RDSTAT);
result = Spi_Read() ;
Spi_Write( 0 ) ; //数据重复输出
MCP2510_CS_H ;
//if( MCP2510_DEBUG ) Uart_Printf( "StatusREG = 0x%x\n", result ) ;
return result;
}
/****************************************************************************
【功能说明】从MCP2510指定地址中读出一个字节
****************************************************************************/
unsigned char MCP2510_Read( U8 address )
{
unsigned char result;
MCP2510_RW_Start() ;
Spi_Write(MCP2510INSTR_READ) ; //0x03
Spi_Write( address ) ;
result = Spi_Read() ;
MCP2510_CS_H ;
return result ;
}
/****************************************************************************
【功能说明】序列读取MCP2510数据
****************************************************************************/
void MCP2510_SRead( U8 address, unsigned char* pdata, U8 nlength )
{
int i;
MCP2510_RW_Start() ;
Spi_Write(MCP2510INSTR_READ);
Spi_Write( address );
for (i=0; i<nlength; i++)
{
*pdata=Spi_Read();
//if( MCP2510_DEBUG ) Uart_Printf( " 0x%x\n", (unsigned char)*pdata ) ;
pdata++;
}
MCP2510_CS_H ;
}
/****************************************************************************
【功能说明】序列写入MCP2510数据
****************************************************************************/
void MCP2510_Swrite( U8 address, unsigned char* pdata, U8 nlength)
{
int i;
MCP2510_RW_Start() ;
Spi_Write(MCP2510INSTR_WRITE);
Spi_Write((unsigned char)address);
for (i=0; i < nlength; i++)
{
Spi_Write( (unsigned char)*pdata );
//if( MCP2510_DEBUG ) Uart_Printf( "0x%x\n", (unsigned char)*pdata ) ;
pdata++;
}
MCP2510_CS_H ;
}
/****************************************************************************
【功能说明】
****************************************************************************/
void MCP2510_SetBandRate(CanBandRate bandrate, int IsBackNormal)
{
U8 value=0;
U8 ReadBackCNT = 0;
// Bit rate calculations.
//
//Input clock fre=16MHz
// In this case, we'll use a speed of 125 kbit/s, 250 kbit/s, 500 kbit/s.
// If we set the length of the propagation segment to 7 bit time quanta,
// and we set both the phase segments to 4 quanta each,
// one bit will be 1+7+4+4 = 16 quanta in length.
//
// setting the prescaler (BRP) to 0 => 500 kbit/s.
// setting the prescaler (BRP) to 1 => 250 kbit/s.
// setting the prescaler (BRP) to 3 => 125 kbit/s.
//
// If we set the length of the propagation segment to 3 bit time quanta,
// and we set both the phase segments to 1 quanta each,
// one bit will be 1+3+2+2 = 8 quanta in length.
// setting the prescaler (BRP) to 0 => 1 Mbit/s.
// Go into configuration mode
MCP2510_Write(MCP2510REG_CANCTRL, MODE_CONFIG);
if( MCP2510_DEBUG ) RETAILMSG(1,(TEXT( "MCP2510REG_CANCTRL = 0x%x\n"), MCP2510_Read(MCP2510REG_CANCTRL) ));
while( ReadBackCNT<8 )
{
value = ( MCP2510_Read( MCP2510REG_CANSTAT ) & 0xe0 );
if(value == MODE_CONFIG ){
//Uart_Printf( "ReadBackCNT = 0x%x\n", ReadBackCNT );
break;
}
ReadBackCNT++ ;
}
if( ReadBackCNT == 8 ) //Set mcp2510's mode failed,redo it again
{
//Uart_Printf( "Set config mode is failed! CANCTRL = 0x%x\n", value );
MCP2510_Reset();
MCP2510_Write(MCP2510REG_CANCTRL, MODE_CONFIG); //redo to set mcp2510 mode
Delay( 150 );
value = ( MCP2510_Read(MCP2510REG_CANCTRL) & 0xe0 ); //read back mode from CANSTAT Register
//Uart_Printf( "Set is 0x%x , Read is 0x%x\n", MODE_CONFIG, value ) ;
}
switch(bandrate){
case BandRate_10kbps:
MCP2510_Write(CNF1, 0x31); //10k 16TQ
MCP2510_Write(CNF2, 0xb0); //PS1=7 TQ PSeg=1 TQ
MCP2510_Write(CNF3, 0x06); //PS2=7 TQ SYNC=1 TQ
//if( MCP2510_DEBUG ) Uart_Printf( "CNF1 = 0x%x\n", MCP2510_Read(CNF1) );
//if( MCP2510_DEBUG ) Uart_Printf( "CNF2 = 0x%x\n", MCP2510_Read(CNF2) );
//if( MCP2510_DEBUG ) Uart_Printf( "CNF3 = 0x%x\n", MCP2510_Read(CNF3) );
break;
case BandRate_125kbps:
MCP2510_Write(CNF1, SJW1|BRP4); //Synchronization Jump Width Length =1 TQ
MCP2510_Write(CNF2, BTLMODE_CNF3|(SEG4<<3)|SEG7); // Phase Seg 1 = 4, Prop Seg = 7
MCP2510_Write(CNF3, SEG4);// Phase Seg 2 = 4
break;
case BandRate_250kbps:
MCP2510_Write(CNF1, SJW1|BRP2); //Synchronization Jump Width Length =1 TQ
MCP2510_Write(CNF2, BTLMODE_CNF3|(SEG4<<3)|SEG7); // Phase Seg 1 = 4, Prop Seg = 7
MCP2510_Write(CNF3, SEG4);// Phase Seg 2 = 4
break;
case BandRate_500kbps:
MCP2510_Write(CNF1, SJW1|BRP1); //Synchronization Jump Width Length =1 TQ
MCP2510_Write(CNF2, BTLMODE_CNF3|(SEG4<<3)|SEG7); // Phase Seg 1 = 4, Prop Seg = 7
MCP2510_Write(CNF3, SEG4);// Phase Seg 2 = 4
break;
case BandRate_1Mbps:
MCP2510_Write(CNF1, SJW1|BRP1); //Synchronization Jump Width Length =1 TQ
MCP2510_Write(CNF2, BTLMODE_CNF3|(SEG3<<3)|SEG2); // Phase Seg 1 = 2, Prop Seg = 3
MCP2510_Write(CNF3, SEG2);// Phase Seg 2 = 1
break;
}
if( IsBackNormal == TRUE )
{
//Enable clock output
MCP2510_Write(CLKCTRL, MODE_NORMAL | CLKEN | CLK8);
}
}
/****************************************************************************
【功能说明】读取MCP2510 CAN总线ID
参数: address为MCP2510寄存器地址
can_id为返回的ID值
返回值
TRUE,表示是扩展ID(29位)
FALSE,表示非扩展ID(11位)
****************************************************************************/
int MCP2510_Read_Can_ID( U8 address, U32* can_id)
{
U32 tbufdata;
unsigned char* p=(unsigned char*)&tbufdata;
MCP2510_SRead(address, p, 4);
*can_id = (tbufdata<<3)|((tbufdata>>13)&0x7);
*can_id &= 0x7ff;
if ( (p[MCP2510LREG_SIDL] & TXB_EXIDE_M) == TXB_EXIDE_M ) {
*can_id = (*can_id<<2) | (p[MCP2510LREG_SIDL] & 0x03);
*can_id <<= 16;
*can_id |= tbufdata>>16;
return TRUE;
}
return FALSE;
}
/***********************************************************\
* 读取MCP2510 接收的数据 *
* 参数: nbuffer为第几个缓冲区可以为3或者4 *
* can_id为返回的ID值 *
* rxRTR表示是否是RXRTR *
* data表示读取的数据 *
* dlc表示data length code *
* 返回值 *
* TRUE,表示是扩展总线 *
* FALSE,表示非扩展总线 *
\***********************************************************/
int MCP2510_Read_Can(U8 nbuffer, int* rxRTR, U32* can_id, U8* data , U8* dlc)
{
U8 mcp_addr = (nbuffer<<4) + 0x31, ctrl;
int IsExt;
IsExt=MCP2510_Read_Can_ID( mcp_addr, can_id);
ctrl=MCP2510_Read(mcp_addr-1);
*dlc=MCP2510_Read( mcp_addr+4);
if ((ctrl & 0x08)) {
*rxRTR = TRUE;
}
else{
*rxRTR = FALSE;
}
*dlc &= DLC_MASK;
//MCP2510_SRead(mcp_addr+5, data, *dlc);
MCP2510_SRead(mcp_addr+5, data, 8);
return IsExt;
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -