stm32制作CAN适配器2--兼容使用周立功上位机

上次我们使用STM32F072制作了一个USB的CAN选配器,可以进行简单的CAN收发功能 ,今天我们利用这个下位机制作一个DLL去替换周立功的ControlCan.dll,这样我们就可以直接使用以前编辑的周立功上位机了。

因为我们的下位机是HID模式,这里就用到了开源的HID上位机源代码

地址:https://github.com/signal11/hidapi

下载后,直接使用里面的hid.c和hidapi.h这两个文件加入到我们的工程中去使用。

1、打开VS,新建项目,这里选择控制台程序

下一步,选择DLL和空项目

2、把hid.c和hidapi.h加入工程,再新建controlcan.h和controlcan.c。

3、其中controlcan.h在周立功提供的库里可以找到,再做一些简单修改

#ifndef CONTROLCAN_H
#define CONTROLCAN_H

//接口卡类型定义
#define VCI_PCI5121         1
#define VCI_PCI9810         2
#define VCI_USBCAN1         3
#define VCI_USBCAN2         4
#define VCI_USBCAN2A        4
#define VCI_PCI9820         5
#define VCI_CAN232          6
#define VCI_PCI5110         7
#define VCI_CANLITE         8
#define VCI_ISA9620         9
#define VCI_ISA5420         10
#define VCI_PC104CAN        11
#define VCI_CANETUDP        12
#define VCI_CANETE          12
#define VCI_DNP9810         13
#define VCI_PCI9840         14
#define VCI_PC104CAN2       15
#define VCI_PCI9820I        16
#define VCI_CANETTCP        17
#define VCI_PEC9920         18
#define VCI_PCIE_9220       18
#define VCI_PCI5010U        19
#define VCI_USBCAN_E_U      20
#define VCI_USBCAN_2E_U     21
#define VCI_PCI5020U        22
#define VCI_EG20T_CAN       23
#define VCI_PCIE9221        24
#define VCI_WIFICAN_TCP     25
#define VCI_WIFICAN_UDP     26
#define VCI_PCIe9120        27
#define VCI_PCIe9110        28
#define VCI_PCIe9140        29
#define VCI_USBCAN_4E_U     31
#define VCI_CANDTU_200UR    32
#define VCI_CANDTU_MINI     33
#define VCI_USBCAN_8E_U     34
#define VCI_CANREPLAY       35
#define VCI_CANDTU_NET      36
#define VCI_CANDTU_100UR    37

//CAN错误码
#define ERR_CAN_OVERFLOW            0x0001  //CAN控制器内部FIFO溢出
#define ERR_CAN_ERRALARM            0x0002  //CAN控制器错误报警
#define ERR_CAN_PASSIVE             0x0004  //CAN控制器消极错误
#define ERR_CAN_LOSE                0x0008  //CAN控制器仲裁丢失
#define ERR_CAN_BUSERR              0x0010  //CAN控制器总线错误
#define ERR_CAN_BUSOFF              0x0020  //总线关闭错误
#define ERR_CAN_BUFFER_OVERFLOW     0x0040  //CAN控制器内部BUFFER溢出
//通用错误码
#define ERR_DEVICEOPENED            0x0100  //设备已经打开
#define ERR_DEVICEOPEN              0x0200  //打开设备错误
#define ERR_DEVICENOTOPEN           0x0400  //设备没有打开
#define ERR_BUFFEROVERFLOW          0x0800  //缓冲区溢出
#define ERR_DEVICENOTEXIST          0x1000  //此设备不存在
#define ERR_LOADKERNELDLL           0x2000  //装载动态库失败
#define ERR_CMDFAILED               0x4000  //执行命令失败错误码
#define ERR_BUFFERCREATE            0x8000  //内存不足

//CANET错误码
#define ERR_CANETE_PORTOPENED       0x00010000  //端口已经被打开
#define ERR_CANETE_INDEXUSED        0x00020000  //设备索引号已经被占用
#define ERR_REF_TYPE_ID             0x00030000  //SetReference或GetReference传递的RefType不存在
#define ERR_CREATE_SOCKET           0x00030002  //创建Socket失败
#define ERR_OPEN_CONNECT            0x00030003  //打开Socket的连接时失败,可能设备连接已经存在
#define ERR_NO_STARTUP              0x00030004  //设备没启动
#define ERR_NO_CONNECTED            0x00030005  //设备无连接
#define ERR_SEND_PARTIAL            0x00030006  //只发送了部分的CAN帧
#define ERR_SEND_TOO_FAST           0x00030007  //数据发得太快,Socket缓冲区满了

//函数调用返回状态值
#define STATUS_OK                   1
#define STATUS_ERR                  0

#define CMD_DESIP                   0
#define CMD_DESPORT                 1
#define CMD_CHGDESIPANDPORT         2
#define CMD_SRCPORT                 2
#define CMD_TCP_TYPE                4  //tcp 工作方式,服务器:1 或是客户端:0
#define TCP_CLIENT                  0
#define TCP_SERVER                  1
//服务器方式下有效
#define CMD_CLIENT_COUNT            5  //连接上的客户端计数
#define CMD_CLIENT                  6  //连接上的客户端
#define CMD_DISCONN_CLINET          7  //断开一个连接
#define CMD_SET_RECONNECT_TIME      8  //使能自动重连
//CANDTU_NET支持GPS
#define CMD_GET_GPS                 9
#define CMD_GET_GPS_NUM             10 //获取GPS信息的数目

typedef unsigned long       DWORD, ULONG;
typedef int                 INT;
typedef void*				HANDLE;
typedef unsigned char       BYTE;
typedef unsigned short      USHORT;
typedef char				CHAR;
typedef unsigned int        UINT;
typedef unsigned char		UCHAR;
typedef unsigned short      UINT16;
typedef void*				PVOID;

typedef struct tagRemoteClient{
    int     iIndex;
    DWORD   port;
    HANDLE  hClient;
    char    szip[32];
}REMOTE_CLIENT;

typedef struct _tagChgDesIPAndPort
{
    char    szpwd[10];
    char    szdesip[20];
    int     desport;
    BYTE    blistenonly;
}CHGDESIPANDPORT;

//1.ZLGCAN系列接口卡信息的数据类型。
typedef  struct  _VCI_BOARD_INFO{
    USHORT  hw_Version;
    USHORT  fw_Version;
    USHORT  dr_Version;
    USHORT  in_Version;
    USHORT  irq_Num;
    BYTE    can_Num;
    CHAR    str_Serial_Num[20];
    CHAR    str_hw_Type[40];
    USHORT  Reserved[4];
} VCI_BOARD_INFO,*PVCI_BOARD_INFO;

//2.定义CAN信息帧的数据类型。
typedef  struct  _VCI_CAN_OBJ{
    UINT    ID;
    UINT    TimeStamp;
    BYTE    TimeFlag;
    BYTE    SendType;
    BYTE    RemoteFlag;//是否是远程帧
    BYTE    ExternFlag;//是否是扩展帧
    BYTE    DataLen;
    BYTE    Data[8];
    BYTE    Reserved[3];    //Reserved[0] 第0位表示特殊的空行或者高亮帧
}VCI_CAN_OBJ,*PVCI_CAN_OBJ;

//3.定义CAN控制器状态的数据类型。
typedef struct _VCI_CAN_STATUS{
    UCHAR   ErrInterrupt;
    UCHAR   regMode;
    UCHAR   regStatus;
    UCHAR   regALCapture;
    UCHAR   regECCapture; 
    UCHAR   regEWLimit;
    UCHAR   regRECounter; 
    UCHAR   regTECounter;
    DWORD   Reserved;
}VCI_CAN_STATUS,*PVCI_CAN_STATUS;

//4.定义错误信息的数据类型。
typedef struct _VCI_ERR_INFO{
    UINT    ErrCode;
    BYTE    Passive_ErrData[3];
    BYTE    ArLost_ErrData;
} VCI_ERR_INFO,*PVCI_ERR_INFO;

//5.定义初始化CAN的数据类型
typedef struct _VCI_INIT_CONFIG{
    DWORD    AccCode;
    DWORD    AccMask;
    DWORD    Reserved;
    UCHAR    Filter;
    UCHAR    Timing0;
    UCHAR    Timing1;
    UCHAR    Mode;
}VCI_INIT_CONFIG,*PVCI_INIT_CONFIG;

/ new add struct for filter /
typedef struct _VCI_FILTER_RECORD{
    DWORD   ExtFrame;   //是否为扩展帧
    DWORD   Start;
    DWORD   End;
}VCI_FILTER_RECORD,*PVCI_FILTER_RECORD;

//定时自动发送帧结构
typedef struct _VCI_AUTO_SEND_OBJ{
    BYTE    Enable;     //使能本条报文 0:禁能 1:使能
    BYTE    Index;      //报文编号     最大支持32条报文
    DWORD   Interval;   //定时发送时间 1ms为单位
    VCI_CAN_OBJ obj;    //报文
}VCI_AUTO_SEND_OBJ,*PVCI_AUTO_SEND_OBJ;

//设置指示灯状态结构
typedef struct _VCI_INDICATE_LIGHT{
    BYTE    Indicate;             //指示灯编号
    BYTE    AttribRedMode:2;      //Red LED灭/亮/闪烁/自控
    BYTE    AttribGreenMode:2;    //Green LED灭/亮/闪烁/自控
    BYTE    AttribReserved:4;     //保留暂时不用
    BYTE    FrequenceRed:2;       //Red LED闪烁频率
    BYTE    FrequenceGreen:2;     //Green LED闪烁频率
    BYTE    FrequenceReserved:4;  //保留暂时不用
} VCI_INDICATE_LIGHT,*PVCI_INDICATE_LIGHT;

//设置转发结构
typedef struct _VCI_CAN_OBJ_REDIRECT{
    BYTE    Action;                //标识开启或停止转发
    BYTE    DestCanIndex;          //CAN目标通道
} VCI_CAN_OBJ_REDIRECT,*PVCI_CAN_OBJ_REDIRECT;

typedef struct _CANDTUTIME {
    UINT16 wYear;
    UINT16 wMonth;
    UINT16 wDay;
    UINT16 wHour;
    UINT16 wMinute;
    UINT16 wSecond;
} CANDTUTIME;

//GPS数据结构
typedef struct _tagCANDTUGPSData
{
    float       fLatitude;  //纬度
    float       fLongitude; //经度
    float       fSpeed;     //速度
    CANDTUTIME  candtuTime;
}CANDTUGPSData, *PCANDTUGPSData;

//获取GPS结构
typedef struct _VCI_CANDTU_GPS_DATA
{
    PCANDTUGPSData pGPSData;    //用户提供接收GPS数据的缓冲区地址
    ULONG          nGPSDataCnt; //可以容纳的GPS数据个数
}VCI_CANDTU_GPS_DATA, *PVCI_CANDTU_GPS_DATA;

#ifdef __cplusplus
#define EXTERNC extern "C" __declspec(dllexport)
#define DEF(a) = a
#else
#define EXTERNC 
#define DEF(a)
#endif

extern "C"
{
EXTERNC DWORD __stdcall VCI_OpenDevice(DWORD DeviceType,DWORD DeviceInd,DWORD Reserved);
EXTERNC DWORD __stdcall VCI_CloseDevice(DWORD DeviceType,DWORD DeviceInd);
EXTERNC DWORD __stdcall VCI_InitCAN(DWORD DeviceType, DWORD DeviceInd, DWORD CANInd, PVCI_INIT_CONFIG pInitConfig);

EXTERNC DWORD __stdcall VCI_ReadBoardInfo(DWORD DeviceType,DWORD DeviceInd,PVCI_BOARD_INFO pInfo);
EXTERNC DWORD __stdcall VCI_ReadErrInfo(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd,PVCI_ERR_INFO pErrInfo);
EXTERNC DWORD __stdcall VCI_ReadCANStatus(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd,PVCI_CAN_STATUS pCANStatus);

EXTERNC DWORD __stdcall VCI_GetReference(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd,DWORD RefType,PVOID pData);
EXTERNC DWORD __stdcall VCI_SetReference(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd,DWORD RefType,PVOID pData);

EXTERNC ULONG __stdcall VCI_GetReceiveNum(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd);
EXTERNC DWORD __stdcall VCI_ClearBuffer(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd);

EXTERNC DWORD __stdcall VCI_StartCAN(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd);
EXTERNC DWORD __stdcall VCI_ResetCAN(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd);

EXTERNC ULONG __stdcall VCI_Transmit(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd,PVCI_CAN_OBJ pSend,ULONG Len);
EXTERNC ULONG __stdcall VCI_Receive(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd,PVCI_CAN_OBJ pReceive,ULONG Len,INT WaitTime DEF(-1));

}

#endif

4、control.c文件,主要是CAN的收发协议,其它的没有进行实现

#include "ControlCAN.h"
#include "hidapi.h"

#ifdef WIN32
#include <windows.h>
#endif

#include <stdio.h>
#include <stdlib.h>

int res;
unsigned char sbuf[65];
unsigned char rbuf[65];
hid_device *handle;
int i;

 DWORD __stdcall VCI_OpenDevice(DWORD DeviceType,DWORD DeviceInd,DWORD Reserved)
 {
	  res = hid_init();
	  if(res == 0)
		return 1;
	  else
		  return 0;
 }

 DWORD __stdcall VCI_CloseDevice(DWORD DeviceType,DWORD DeviceInd)
 {
	 hid_close(handle);
	 res = hid_exit();

	 return 1;
 }
 DWORD __stdcall VCI_InitCAN(DWORD DeviceType, DWORD DeviceInd, DWORD CANInd, PVCI_INIT_CONFIG pInitConfig)
 {
	handle = hid_open(0x1234, 0x5678, NULL);

	if(handle == 0)
		return 0;
	else
		return 1;
 }

 DWORD __stdcall VCI_ReadBoardInfo(DWORD DeviceType,DWORD DeviceInd,PVCI_BOARD_INFO pInfo)
 {
	 return 1;
 }
 DWORD __stdcall VCI_ReadErrInfo(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd,PVCI_ERR_INFO pErrInfo)
 {
	 return 1;
 }
 DWORD __stdcall VCI_ReadCANStatus(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd,PVCI_CAN_STATUS pCANStatus)
 {
	 return 1;
 }

 DWORD __stdcall VCI_GetReference(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd,DWORD RefType,PVOID pData)
 {
	
	 return 1;
 }
 DWORD __stdcall VCI_SetReference(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd,DWORD RefType,PVOID pData)
 {
	 return 1;
 }

 ULONG __stdcall VCI_GetReceiveNum(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd)
 {
	// res = hid_read_timeout(handle, rbuf, 20,0);
	 res = 20;
	 if(res == 20)
		return 1;
	 else
		 return 0;
 }
 DWORD __stdcall VCI_ClearBuffer(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd)
 {
	 return 1;
 }

 DWORD __stdcall VCI_StartCAN(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd)
 {
	 if(handle == 0)
		return 0;

	 hid_set_nonblocking(handle,1);

	 return 1;
 }
 DWORD __stdcall VCI_ResetCAN(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd)
 {
	 return 1;
 }

 ULONG __stdcall VCI_Transmit(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd,PVCI_CAN_OBJ pSend,ULONG Len)
 {
	 if(handle == 0)
		return 0;

	 sbuf[0] = 0;
	 sbuf[5] = pSend->ExternFlag == 0?0:4;
	 sbuf[6] = pSend->RemoteFlag==0?0:2;

	 sbuf[7] = (byte)pSend->ID;
	 sbuf[8] = (byte)(pSend->ID >> 8);
	 sbuf[9] = (byte)(pSend->ID >> 16);
	 sbuf[10] = (byte)(pSend->ID >> 24);
	 sbuf[11] =  pSend->DataLen;

	for(i = 0;i<8;i++)
	 {
		 sbuf[12+i] = pSend->Data[i];

	}
	res = hid_write(handle, sbuf, 21);	

	 return 1;
 }
 ULONG __stdcall VCI_Receive(DWORD DeviceType,DWORD DeviceInd,DWORD CANInd,PVCI_CAN_OBJ pReceive,ULONG Len,INT WaitTime)
 {
	  //res = hid_read_timeout(handle, rbuf, 20,0);
	 res = hid_read(handle, rbuf, 20);

	  if(res == 20)
	  {
		 UINT id = ((UINT)rbuf[9] << 24) + ((UINT)rbuf[8] << 16) + ((UINT)rbuf[7] << 8) + rbuf[6];


		 pReceive->ID = id;
		 pReceive->ExternFlag = rbuf[4] == 0?0:1;
		 pReceive->RemoteFlag = rbuf[5] == 0?0:1;
		 pReceive->DataLen = rbuf[10];

		 for(i = 0;i<8;i++)
		 {
			 pReceive->Data[i] = rbuf[i+11];

		 }

		 return 1;
	  }
	  else
	  {
		  return 0;
	  }
 }

5、进行工程编译,生成的ControlCAN.dll,去替换以前写的周立功上位机,就可以使用了

 

  • 7
    点赞
  • 34
    收藏
    觉得还不错? 一键收藏
  • 7
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值