• 使用USBCAN通讯


    使用ZLG USBCAN-E-U,通过C#实现通讯:

    1、建一个CANDevice实体类,传递一些参数,并且保存一些设备的型号、ID,以及波特率等参数

    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Text;
    using System.Threading.Tasks;
    using System.Threading;
    using System.Runtime.InteropServices;
    
    namespace ZLGUSBCAN
    {
        [Serializable]
        public class CardType
        {
            public string TyPeName { get; set; }
            public uint TypeId { get; set; }
        }
        [Serializable]
        public class BoudRoute
        {
            public string RateName { get; set; }
            public UInt32 RateValue { get; set; }
        }
    
        [Serializable]
        public class CANDevice
        {        
            //private readonly uint devType;
            //private readonly uint devID;
            //private readonly uint canID;
            //private readonly uint baudRate;
            //private readonly uint sendTimeout;
    
            public uint DevType { get; set; }
            public uint DevID { get; set; }
            public uint CANID { get; set; }
            public uint BoudRate { get; set; }
            public uint SendTimeout { get; set; }
            //CAN参数
            public uint AccCode { get; set; }
            public uint AccMask { get; set; }
            public uint Reserved { get; set; }
            public byte Filter { get; set; }
            public byte Timing0 { get; set; }
            public byte Timing1 { get; set; }
            public byte Mode { get; set; }
            public List<CardType> CardTypeList { get; set; }
            public List<BoudRoute> BoudRateList { get; set; }
    
            public CANDevice(uint devType, uint devID, uint boudRate, uint sendTimeout, uint canID = 0)
            {
                this.DevType = devType;
                this.DevID = devID;
                this.CANID = canID;
                this.BoudRate = boudRate;
                this.SendTimeout = sendTimeout;
    
            }
    
            public CANDevice()
            {
                this.CardTypeList = new List<CardType>();           
                this.CardTypeList.Add(new CardType { TyPeName = "VCI_USBCAN_E_U", TypeId = 20 });
                this.CardTypeList.Add(new CardType { TyPeName = "VCI_USBCAN_2E_U", TypeId = 21 });
    
                this.BoudRateList = new List<BoudRoute>();
                this.BoudRateList.Add(new BoudRoute { RateName = "1000Kbps", RateValue = 0x060003 });
                this.BoudRateList.Add(new BoudRoute { RateName = "800Kbps", RateValue = 0x060004 });
                this.BoudRateList.Add(new BoudRoute { RateName = "500Kbps", RateValue = 0x060007 });
                this.BoudRateList.Add(new BoudRoute { RateName = "250Kbps", RateValue = 0x1C0008 });
                this.BoudRateList.Add(new BoudRoute { RateName = "125Kbps", RateValue = 0x1C0011 });
                this.BoudRateList.Add(new BoudRoute { RateName = "100Kbps", RateValue = 0x160023 });
                this.BoudRateList.Add(new BoudRoute { RateName = "50Kbps", RateValue = 0x1C002C });
                this.BoudRateList.Add(new BoudRoute { RateName = "20Kbps", RateValue = 0x1600B3 });
                this.BoudRateList.Add(new BoudRoute { RateName = "10Kbps", RateValue = 0x1C00E0 });
                this.BoudRateList.Add(new BoudRoute { RateName = "5Kbps", RateValue = 0x1C01C1 });           
            }   
        }
    }
    View Code

    2、建一个ControlCAN,对周立功提供的接口函数进行二次开发

    首先需要引入设备供应商提供的接口函数库,在C#中属于非托管,因此需要通过[DllImport("路径.dll")]这样的方式来引入。特别注意:该接口函数如果是32位,注意你的解决方案平台需要时X86;还有,你要将所有引用的文件都复制到路径下。

    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Text;
    using System.Threading.Tasks;
    using System.Threading;
    using System.Runtime.InteropServices;
    using System.Runtime.Versioning;
    using Microsoft.Win32.SafeHandles;
    
    namespace ZLGUSBCAN
    {
        public class _VCI_CAN_OBJ
        {
            public uint ID { get; set; }
            public uint TimeStamp { get; set; }
            public byte TimeFlag { get; set; }
            /// <summary>
            /// 发送格式:0:正常发送 1:单次正常发送 2:自发自收 3.单次自发自收
            /// </summary>
            public byte SendType { get; set; }
            /// <summary>
            /// 帧格式:0:数据帧 1:远程帧
            /// </summary>
            public byte RemoteFlag { get; set; }
            /// <summary>
            /// 帧类型:0:标准帧 1为扩展帧,29位ID
            /// </summary>
            public byte ExternFlag { get; set; }
            public byte DataLen { get; set; }
            public byte[] Data { get; set; }
            public byte[] Reserved { get; set; }
        }
    
        public class _VCI_INIT_CONFIG
        {
            public uint AccCode { get; set; }
            public uint AccMask { get; set; }
            public byte Reserved { get; set; }
            public byte Filter { get; set; }
            public byte Timing0 { get; set; }
            public byte Timing1 { get; set; }
            public byte Mode { get; set; }
        }
        public class ControlCAN
        {
            #region 接口函数定义
            [StructLayoutAttribute(LayoutKind.Sequential)]
            struct VCI_CAN_OBJ
            {
                public uint ID;
                public uint TimeStamp;
                public byte TimeFlag;
                public byte SendType;//发送格式:0:正常发送 1:单次正常发送 2:自发自收 3.单次自发自收
                public byte RemoteFlag;//帧格式:0:数据帧 1:远程帧
                public byte ExternFlag;//帧类型:0:标准帧 1为扩展帧,29位ID
                public byte DataLen;
                [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 8)]
                public byte[] Data;
                [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 3)]
                public byte[] Reserved;
            }
    
            [StructLayoutAttribute(LayoutKind.Sequential)]
            struct VCI_INIT_CONFIG
            {
                public uint AccCode;
                public uint AccMask;
                public uint Reserved;
                public byte Filter;
                public byte Timing0;
                public byte Timing1;
                public byte Mode;
            }
    
            [StructLayoutAttribute(LayoutKind.Sequential)]
            struct VCI_FILTER_RECORD
            {
                public uint ExtFrame;
                public uint Start;
                public uint End;
            }
    
            [DllImport("kernel32.dll")]
            static extern uint GetLastError();
    
            [DllImport("ControlCAN.dll")]
            static extern uint VCI_OpenDevice(uint DevType, uint DevIndex, uint Reserved);//Reserved系统保留字段
    
            [DllImport("ControlCAN.dll")]
            static extern uint VCI_CloseDevice(uint DevType, uint DevIndex);
    
            [DllImport("ControlCAN.dll")]
            static extern uint VCI_StartCAN(uint DevType, uint DevIndex, uint CANIndex);
    
            [DllImport("ControlCAN.dll")]
            static extern uint VCI_ResetCAN(uint DevType, uint DevIndex, uint CANIndex);
    
            [DllImport("ControlCAN.dll")]
            static extern uint VCI_SetReference(uint DevType, uint DevIndex, uint CANIndex, uint RefType, IntPtr pData);
    
            //[DllImport("ControlCAN.dll")]
            //unsafe static extern UInt32 VCI_SetReference(UInt32 DeviceType, UInt32 DeviceInd, UInt32 CANInd, UInt32 RefType, byte* pData);
    
            [DllImport("ControlCAN.dll")]
            static extern uint VCI_Receive(uint DevType, uint DevIndex, uint CANIndex, [Out] VCI_CAN_OBJ[] pReceive, uint Len, int WaitTime);
            //参照官方
            [DllImport("ControlCAN.dll", CharSet = CharSet.Ansi)]
            static extern UInt32 VCI_Receive(UInt32 DeviceType, UInt32 DeviceInd, UInt32 CANInd, IntPtr pReceive, UInt32 Len, Int32 WaitTime);
    
            [DllImport("ControlCAN.dll")]
            static extern uint VCI_Transmit(uint DevType, uint DevIndex, uint CANIndex, [In] VCI_CAN_OBJ[] pSend, uint Len);
    
            [DllImport("ControlCAN.dll")]
            static extern uint VCI_InitCAN(uint DevType, uint DevIndex, uint CANIndex, ref VCI_INIT_CONFIG pInitConfig);
    
            [DllImport("ControlCAN.dll")]
            static extern uint VCI_GetReceiveNum(uint DevType, uint DevIndex, uint CANIndex);
    
            [DllImport("ControlCAN.dll")]
            static extern uint VCI_ClearBuffer(uint DevType, uint DevIndex, uint CANIndex);
    
            [DllImport("RC500USB.dll")]
            static extern byte RC500USB_init();
    
            [DllImport("RC500USB.dll")]
            static extern void RC500USB_exit();
    
            [DllImport("RC500USB.dll")]
            static extern byte RC500USB_request(byte mode, ref ushort tagtype);
    
            [DllImport("RC500USB.dll")]
            static extern byte RC500USB_anticoll(byte bcnt, ref uint snr);
    
            [DllImport("RC500USB.dll")]
            static extern byte RC500USB_select(uint snr, ref byte size);
    
            [DllImport("RC500USB.dll")]
            internal static extern byte RC500USB_authkey(byte mode, [In] byte[] key, byte secnr);
    
            [DllImport("RC500USB.dll")]
            internal static extern byte RC500USB_read(byte addr, [Out] byte[] data);
    
            [DllImport("RC500USB.dll")]
            internal static extern byte RC500USB_buzzer(byte contrl, byte opentm, byte closetm, byte repcnt);
    
            // serialport
            //Win32 io errors
            public const int ERROR_BROKEN_PIPE = 109;
            public const int ERROR_NO_DATA = 232;
            public const int ERROR_HANDLE_EOF = 38;
            public const int ERROR_IO_INCOMPLETE = 996;
            public const int ERROR_IO_PENDING = 997;
            public const int ERROR_FILE_EXISTS = 0x50;
            public const int ERROR_FILENAME_EXCED_RANGE = 0xCE;  // filename too long.
            public const int ERROR_MORE_DATA = 234;
            public const int ERROR_CANCELLED = 1223;
            public const int ERROR_FILE_NOT_FOUND = 2;
            public const int ERROR_PATH_NOT_FOUND = 3;
            public const int ERROR_ACCESS_DENIED = 5;
            public const int ERROR_INVALID_HANDLE = 6;
            public const int ERROR_NOT_ENOUGH_MEMORY = 8;
            public const int ERROR_BAD_COMMAND = 22;
            public const int ERROR_SHARING_VIOLATION = 32;
            public const int ERROR_OPERATION_ABORTED = 995;
            public const int ERROR_NO_ASSOCIATION = 1155;
            public const int ERROR_DLL_NOT_FOUND = 1157;
            public const int ERROR_DDE_FAIL = 1156;
            public const int ERROR_INVALID_PARAMETER = 87;
            public const int ERROR_PARTIAL_COPY = 299;
    
            // Since C# does not provide access to bitfields and the native DCB structure contains
            // a very necessary one, these are the positional offsets (from the right) of areas
            // of the 32-bit integer used in SerialStream's SetDcbFlag() and GetDcbFlag() methods.
            internal const int FBINARY = 0;
            internal const int FPARITY = 1;
            internal const int FOUTXCTSFLOW = 2;
            internal const int FOUTXDSRFLOW = 3;
            internal const int FDTRCONTROL = 4;
            internal const int FDSRSENSITIVITY = 6;
            internal const int FTXCONTINUEONXOFF = 7;
            internal const int FOUTX = 8;
            internal const int FINX = 9;
            internal const int FERRORCHAR = 10;
            internal const int FNULL = 11;
            internal const int FRTSCONTROL = 12;
            internal const int FABORTONOERROR = 14;
            internal const int FDUMMY2 = 15;
    
            // The following are unique to the SerialPort/SerialStream classes
            internal const byte ONESTOPBIT = 0;
            internal const byte ONE5STOPBITS = 1;
            internal const byte TWOSTOPBITS = 2;
    
            public const int FILE_READ_DATA = (0x0001),
            FILE_LIST_DIRECTORY = (0x0001),
            FILE_WRITE_DATA = (0x0002),
            FILE_ADD_FILE = (0x0002),
            FILE_APPEND_DATA = (0x0004),
            FILE_ADD_SUBDIRECTORY = (0x0004),
            FILE_CREATE_PIPE_INSTANCE = (0x0004),
            FILE_READ_EA = (0x0008),
            FILE_WRITE_EA = (0x0010),
            FILE_EXECUTE = (0x0020),
            FILE_TRAVERSE = (0x0020),
            FILE_DELETE_CHILD = (0x0040),
            FILE_READ_ATTRIBUTES = (0x0080),
            FILE_WRITE_ATTRIBUTES = (0x0100),
            FILE_SHARE_READ = 0x00000001,
            FILE_SHARE_WRITE = 0x00000002,
            FILE_SHARE_DELETE = 0x00000004,
            FILE_ATTRIBUTE_READONLY = 0x00000001,
            FILE_ATTRIBUTE_HIDDEN = 0x00000002,
            FILE_ATTRIBUTE_SYSTEM = 0x00000004,
            FILE_ATTRIBUTE_DIRECTORY = 0x00000010,
            FILE_ATTRIBUTE_ARCHIVE = 0x00000020,
            FILE_ATTRIBUTE_NORMAL = 0x00000080,
            FILE_ATTRIBUTE_TEMPORARY = 0x00000100,
            FILE_ATTRIBUTE_COMPRESSED = 0x00000800,
            FILE_ATTRIBUTE_OFFLINE = 0x00001000,
            FILE_NOTIFY_CHANGE_FILE_NAME = 0x00000001,
            FILE_NOTIFY_CHANGE_DIR_NAME = 0x00000002,
            FILE_NOTIFY_CHANGE_ATTRIBUTES = 0x00000004,
            FILE_NOTIFY_CHANGE_SIZE = 0x00000008,
            FILE_NOTIFY_CHANGE_LAST_WRITE = 0x00000010,
            FILE_NOTIFY_CHANGE_LAST_ACCESS = 0x00000020,
            FILE_NOTIFY_CHANGE_CREATION = 0x00000040,
            FILE_NOTIFY_CHANGE_SECURITY = 0x00000100,
            FILE_ACTION_ADDED = 0x00000001,
            FILE_ACTION_REMOVED = 0x00000002,
            FILE_ACTION_MODIFIED = 0x00000003,
            FILE_ACTION_RENAMED_OLD_NAME = 0x00000004,
            FILE_ACTION_RENAMED_NEW_NAME = 0x00000005,
            FILE_CASE_SENSITIVE_SEARCH = 0x00000001,
            FILE_CASE_PRESERVED_NAMES = 0x00000002,
            FILE_UNICODE_ON_DISK = 0x00000004,
            FILE_PERSISTENT_ACLS = 0x00000008,
            FILE_FILE_COMPRESSION = 0x00000010,
            OPEN_EXISTING = 3,
            OPEN_ALWAYS = 4,
            FILE_FLAG_WRITE_THROUGH = unchecked((int)0x80000000),
            FILE_FLAG_OVERLAPPED = 0x40000000,
            FILE_FLAG_NO_BUFFERING = 0x20000000,
            FILE_FLAG_RANDOM_ACCESS = 0x10000000,
            FILE_FLAG_SEQUENTIAL_SCAN = 0x08000000,
            FILE_FLAG_DELETE_ON_CLOSE = 0x04000000,
            FILE_FLAG_BACKUP_SEMANTICS = 0x02000000,
            FILE_FLAG_POSIX_SEMANTICS = 0x01000000,
            FILE_TYPE_UNKNOWN = 0x0000,
            FILE_TYPE_DISK = 0x0001,
            FILE_TYPE_CHAR = 0x0002,
            FILE_TYPE_PIPE = 0x0003,
            FILE_TYPE_REMOTE = unchecked((int)0x8000),
            FILE_VOLUME_IS_COMPRESSED = 0x00008000;
    
            // The following are unique to the SerialPort/SerialStream classes
            internal const int DTR_CONTROL_DISABLE = 0x00;
            internal const int DTR_CONTROL_ENABLE = 0x01;
            internal const int DTR_CONTROL_HANDSHAKE = 0x02;
    
            internal const int RTS_CONTROL_DISABLE = 0x00;
            internal const int RTS_CONTROL_ENABLE = 0x01;
            internal const int RTS_CONTROL_HANDSHAKE = 0x02;
            internal const int RTS_CONTROL_TOGGLE = 0x03;
    
            internal const int MS_CTS_ON = 0x10;
            internal const int MS_DSR_ON = 0x20;
    
            internal const byte DEFAULTXONCHAR = (byte)17;
            internal const byte DEFAULTXOFFCHAR = (byte)19;
    
            internal const byte EOFCHAR = (byte)26;
    
            [StructLayoutAttribute(LayoutKind.Sequential)]
            public struct COMMTIMEOUTS
            {
                public int ReadIntervalTimeout;
                public int ReadTotalTimeoutMultiplier;
                public int ReadTotalTimeoutConstant;
                public int WriteTotalTimeoutMultiplier;
                public int WriteTotalTimeoutConstant;
            }
    
            [StructLayout(LayoutKind.Sequential)]
            internal struct DCB
            {
                public uint DCBlength;
                public uint BaudRate;
                public uint Flags;
                public ushort wReserved;
                public ushort XonLim;
                public ushort XoffLim;
                public byte ByteSize;
                public byte Parity;
                public byte StopBits;
                public byte XonChar;
                public byte XoffChar;
                public byte ErrorChar;
                public byte EofChar;
                public byte EvtChar;
                public ushort wReserved1;
            }
    
            public const int GENERIC_READ = unchecked(((int)0x80000000));
            public const int GENERIC_WRITE = (0x40000000);
    
            internal const int PURGE_TXABORT = 0x0001;  // Kill the pending/current writes to the comm port.
            internal const int PURGE_RXABORT = 0x0002;  // Kill the pending/current reads to the comm port.
            internal const int PURGE_TXCLEAR = 0x0004;  // Kill the transmit queue if there.
            internal const int PURGE_RXCLEAR = 0x0008;  // Kill the typeahead buffer if there.
            #endregion
    
            private static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
    
            public uint MaxFrames { get; set; } = 16;
            public int ReceiveTimeout { get; set; } = 2000;
    
            private int disposed = 0;
            private CANDevice objCANDevice;
    
            protected virtual void Dispose(bool disposing)
            {
                if (Interlocked.Exchange(ref disposed, 1) == 0)
                {
                    if (Interlocked.Exchange(ref disposed, 1) == 0)
                    {
                        //释放非托管资源
                        if (ControlCAN.VCI_CloseDevice(objCANDevice.DevType, objCANDevice.DevID) == 0)
                        {
                            log.Error("Failed to close CAN device");
                        }
                    }
                }
            }
    
            public void Dispose()
            {
                ///<summary>
                /// 实现IDisposable中的Dispose方法
                ///</summary>
                Dispose(true);//必须为true
                //通知垃圾回收机制不再调用终结器(析构器)
                GC.SuppressFinalize(this);//禁止终结操作
            }
    
            /// <summary>
            /// 
            /// </summary>
            /// <param name="devType"></param>
            /// <param name="devID"></param>
            /// <param name="baudRate"></param>
            /// <param name="sendTimeout"></param>
            /// <param name="canID">默认为0</param>
            public ControlCAN(CANDevice canDevice)
            {
                objCANDevice = canDevice;
            }
    
            /// <summary>
            /// 打开CAN设备
            /// </summary>
            /// <returns></returns>
            public bool OpenCAN()
            {
                if (VCI_OpenDevice(objCANDevice.DevType, objCANDevice.DevID, 0) == 0)
                {
                    //log.Error("Failed to open CAN device");
                    //uint error = GetLastError();
                    return false;
                }
                return true;
            }
    
            /// <summary>
            /// 启动某一路CAN
            /// </summary>
            public bool StartCan(uint canId = 0)
            {
                if (Thread.VolatileRead(ref disposed) != 0)
                {
                    throw new ObjectDisposedException("CanDevice already disposed");
                }
    
                if (canId != 0)
                {
                    objCANDevice.CANID = canId;
                }
    
                if (!SetBaudRate(objCANDevice.BoudRate))
                {
                    throw new Exception("波特率设置失败!");
                }
                if (!SetWorkingMode())
                {
                    throw new Exception("工作模式设置失败!");
                }
    
                if (VCI_StartCAN(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID) == 0)
                {
    
                    //log.Error("Failed to start CAN device");
                    return false;
                }
                if (!SetSendTimeout())
                {
                    throw new Exception("发送超时设置失败!");
                }
                ClearBuffer();
    
                return true;
            }
    
            /// <summary>
            /// 设置波特率
            /// </summary>
            /// <param name="baudRate">uint类型</param>
            /// <returns></returns>
            public bool SetBaudRate(uint baudRate)
            {
                if (Thread.VolatileRead(ref disposed) != 0)
                {
                    throw new ObjectDisposedException("CanDevice already disposed");
                }
    
                IntPtr ptr = Marshal.AllocHGlobal(Marshal.SizeOf(baudRate));
    
                try
                {
                    Marshal.WriteInt32(ptr, (int)baudRate);
                    if (VCI_SetReference(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID, 0, ptr) == 0)
                    {
                        log.Error("Failed to set CAN baud rate");
                        return false;
                    }
                }
                finally
                {
                    Marshal.FreeHGlobal(ptr);
                }
    
                return true;
            }
    
            /// <summary>
            /// 设置工作模式,需提供工作模式Id
            /// 必须先设置波特率再设置工模式
            /// </summary>
            /// <param name="modeId"> =0 表示正常模式(相当于正常节点), =1 表示只听模式(只接收,不影响总线)</param>
            /// <returns></returns>
            public bool SetWorkingMode(byte modeId = 0)
            {
                if (Thread.VolatileRead(ref disposed) != 0)
                {
                    throw new ObjectDisposedException("CanDevice already disposed");
                }
    
                VCI_INIT_CONFIG initConfig = new VCI_INIT_CONFIG();
                initConfig.Mode = modeId;//正常模式
    
                if (VCI_InitCAN(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID, ref initConfig) == 0)
                {
                    log.Error("Failed to set CAN working mode");
                    return false;
                }
    
                return true;
            }
    
            /// <summary>
            ///设置发送超时时间
            /// </summary>
            public bool SetSendTimeout(uint timeout = 2000)
            {
                if (Thread.VolatileRead(ref disposed) != 0)
                {
                    throw new ObjectDisposedException("CanDevice already disposed");
                }
    
                IntPtr ptr = Marshal.AllocHGlobal(Marshal.SizeOf(timeout));
    
                try
                {
                    Marshal.WriteInt32(ptr, (int)timeout);
                    if (VCI_SetReference(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID, 4, ptr) == 0)
                    {
                        log.Error("Failed to set CAN send timeout");
                        return false;
                    }
                }
                finally
                {
                    Marshal.FreeHGlobal(ptr);
                }
    
                return true;
            }
    
            /// <summary>
            /// 发送数据
            /// </summary>
            /// <param name="frames">帧结构体数组</param>
            /// 举例:VCI_CAN_OBJ[] frames=new VCI_CAN_OBJ[2];//将发送两帧数据
            /// frames[0].ID=0x00000001;//第一帧ID
            /// frames[0].SendType=0;//正常发送
            /// frames[0].RemoteFlag=0;//数据帧
            /// frames[0].ExternFlag=0;//标准帧
            /// frames[0].DataLen=1;//数据长度
            /// frames[0].Data[0]=0x56;//数据
            /// frames[1]~
            /// <returns></returns>
            public bool Transmit(_VCI_CAN_OBJ[] frames)
            {
                if (Thread.VolatileRead(ref disposed) != 0)
                {
                    throw new ObjectDisposedException("CanDevice already disposed");
                }
                int length = frames.Length;
                VCI_CAN_OBJ[] obj = new VCI_CAN_OBJ[length];
                for (int i = 0; i < length; i++)
                {
                    obj[i] = new VCI_CAN_OBJ
                    {
                        ID = frames[i].ID,
                        TimeFlag = frames[i].TimeFlag,
                        TimeStamp = frames[i].TimeStamp,
                        SendType = frames[i].SendType,
                        RemoteFlag = frames[i].RemoteFlag,
                        ExternFlag = frames[i].ExternFlag,
                        Data = frames[i].Data,
                        DataLen = frames[i].DataLen,
                        Reserved = frames[i].Reserved
                    };
                }
    
                return VCI_Transmit(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID, obj, (uint)length) != 0;
            }
    
            public bool Transmit(_VCI_CAN_OBJ frame)
            {
                if (Thread.VolatileRead(ref disposed) != 0)
                {
                    throw new ObjectDisposedException("CanDevice already disposed");
                }
    
                VCI_CAN_OBJ[] frames = new VCI_CAN_OBJ[1];
                frames[0] = new VCI_CAN_OBJ
                {
                    ID = frame.ID,
                    TimeFlag = frame.TimeFlag,
                    TimeStamp = frame.TimeStamp,
                    SendType = frame.SendType,
                    RemoteFlag = frame.RemoteFlag,
                    ExternFlag = frame.ExternFlag,
                    Data = frame.Data,
                    DataLen = frame.DataLen,
                    Reserved = frame.Reserved
                };
    
                return VCI_Transmit(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID, frames, (uint)frames.Length) != 0;
            }
    
            /// <summary>
            /// 设置CAN相关参数
            /// </summary>
            /// <param name="config"></param>
            /// <returns></returns>
            public bool InitCAN(_VCI_INIT_CONFIG config)
            {
                if (Thread.VolatileRead(ref disposed) != 0)
                {
                    throw new ObjectDisposedException("CanDevice already disposed");
                }
    
                VCI_INIT_CONFIG obj = new VCI_INIT_CONFIG
                {
                    AccCode = config.AccCode,
                    AccMask = config.AccMask,
                    Mode = config.Mode,
                    Filter = config.Filter,
                    Timing0 = config.Timing0,
                    Timing1 = config.Timing1,
                    Reserved = config.Reserved
                };
    
                if (VCI_InitCAN(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID, ref obj) == 0)
                {
                    return false;
                }
                return true;
            }
    
            #region 参考ZLG官方示例
            /// <summary>
            /// 接收数据
            /// </summary>
            /// <returns></returns>
            public List<_VCI_CAN_OBJ> Receive()
            {
                //获取CAN通道缓冲区中已接收但未读取的帧数量
                uint receiveNum = VCI_GetReceiveNum(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID);
                if (receiveNum == 0) return null;
    
                //如果缓冲区未读取的帧数量大于规定的最大读取量,只能读取规定的最大量的帧数据,其它数据丢失
                uint needReceiveNum = receiveNum > MaxFrames ? MaxFrames : receiveNum;
    
                IntPtr pt = Marshal.AllocHGlobal(Marshal.SizeOf(typeof(VCI_CAN_OBJ)) * (Int32)receiveNum);
                //返回值为实际读取的帧数量,数据填充至buf
                receiveNum = VCI_Receive(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID, pt, receiveNum, 100);
    
                List<_VCI_CAN_OBJ> result = new List<_VCI_CAN_OBJ>();
                //string str = "";
                for (int i = 0; i < receiveNum; i++)
                {
                    VCI_CAN_OBJ obj = (VCI_CAN_OBJ)Marshal.PtrToStructure((IntPtr)((UInt32)pt + i * Marshal.SizeOf(typeof(VCI_CAN_OBJ))), typeof(VCI_CAN_OBJ));
    
                   
                    //foreach (var item in obj.Data)
                    //{
                    //    str += " " + System.Convert.ToString(item, 16);
                    //}
    
                    result.Add(new _VCI_CAN_OBJ
                    {
                        ID = obj.ID,
                        TimeFlag = obj.TimeFlag,
                        TimeStamp = obj.TimeStamp,
                        SendType = obj.SendType,
                        RemoteFlag = obj.RemoteFlag,
                        ExternFlag = obj.ExternFlag,
                        Data = obj.Data,
                        DataLen = obj.DataLen,
                        Reserved = obj.Reserved
                    });
                }
                return result;
            }
    
            /// <summary>
            /// 直接获取接收数据的string形式
            /// </summary>
            /// <returns></returns>
            public string OnlyReceiveData()
            {
                //获取CAN通道缓冲区中已接收但未读取的帧数量
                uint receiveNum = VCI_GetReceiveNum(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID);
                if (receiveNum == 0) return null;
    
                //如果缓冲区未读取的帧数量大于规定的最大读取量,只能读取规定的最大量的帧数据,其它数据丢失
                uint needReceiveNum = receiveNum > MaxFrames ? MaxFrames : receiveNum;
    
                IntPtr pt = Marshal.AllocHGlobal(Marshal.SizeOf(typeof(VCI_CAN_OBJ)) * (Int32)receiveNum);
                //返回值为实际读取的帧数量,数据填充至buf
                receiveNum = VCI_Receive(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID, pt, receiveNum, 100);
               
                string str = "";
                for (int i = 0; i < receiveNum; i++)
                {
                    VCI_CAN_OBJ obj = (VCI_CAN_OBJ)Marshal.PtrToStructure((IntPtr)((UInt32)pt + i * Marshal.SizeOf(typeof(VCI_CAN_OBJ))), typeof(VCI_CAN_OBJ));
    
                    foreach (var item in obj.Data)
                    {
                        str += " " + System.Convert.ToString(item, 16);
                    }
                }
                return str;
            }
            #endregion
    
            #region 修改接收方法:调用的VCI_Receive参数不同
            /// <summary>
            /// 
            /// </summary>
            /// <returns></returns>
            public List<_VCI_CAN_OBJ> _Receive()
            {
                if (Thread.VolatileRead(ref disposed) != 0)
                {
                    throw new ObjectDisposedException("CanDevice already disposed");
                }
    
                VCI_CAN_OBJ[] buf = new VCI_CAN_OBJ[MaxFrames];//MaxFrames为规定的最大接收数
    
                uint num = VCI_Receive(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID, buf, (uint)buf.Length,2000);
               
               VCI_CAN_OBJ[] frames = new VCI_CAN_OBJ[num];
                Array.Copy(buf, frames, num);
    
                List<_VCI_CAN_OBJ> result = new List<_VCI_CAN_OBJ>();
                foreach (var obj in frames)
                {
                    result.Add(new _VCI_CAN_OBJ
                    {
                        ID = obj.ID,
                        TimeFlag = obj.TimeFlag,
                        TimeStamp = obj.TimeStamp,
                        SendType = obj.SendType,
                        RemoteFlag = obj.RemoteFlag,
                        ExternFlag = obj.ExternFlag,
                        Data = obj.Data,
                        DataLen = obj.DataLen,
                        Reserved = obj.Reserved
                    });
                }
    
                return result;
            }
            #endregion
            /// <summary>
            /// 查看缓存区是否存在未读取数据
            /// </summary>
            /// <returns></returns>
            public bool HasReceive()
            {
                return VCI_GetReceiveNum(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID) != 0;
            }
    
            /// <summary>
            /// 清除接收缓冲区数据
            /// </summary>
            /// <returns></returns>
            public bool ClearBuffer()
            {
                if (Thread.VolatileRead(ref disposed) != 0)
                {
                    throw new ObjectDisposedException("CanDevice already disposed");
                }
    
                if (ControlCAN.VCI_ClearBuffer(objCANDevice.DevType, objCANDevice.DevID, objCANDevice.CANID) == 0)
                {
                    log.Error("Failed to clear can.");
                    return false;
                }
    
                return true;
            }
    
            /// <summary>
            /// 关闭CAN
            /// </summary>
            /// <returns></returns>
            public bool CloseCAN()
            {
                if (VCI_CloseDevice(objCANDevice.DevType, objCANDevice.DevID) == 0)
                {
                    //log.Error("Failed to close CAN device");
                    return false;
                }
                return true;
            }
        }
    }
    View Code

    基于提供的接口函数,对连接、打开、发送、接收等方法进行再次封装。

    3、这样就可以实现通讯了

    我做了一个简单的winform界面,可以进行简单的参数设置,并且能发送、接收数据。

    using System;
    using System.Collections.Generic;
    using System.ComponentModel;
    using System.Data;
    using System.Drawing;
    using System.Linq;
    using System.Text;
    using System.Threading.Tasks;
    using System.Windows.Forms;
    
    namespace ZLGUSBCAN
    {
        public partial class Form1 : Form
        {
            private CANDevice objCANDevice = null;
            private ControlCAN objControlCAN = null;
            private bool isOpen = false;
    
            public Form1()
            {
                InitializeComponent();
    
                this.cboDevType.DataSource = new CANDevice().CardTypeList;
                this.cboDevType.ValueMember = "TypeId";
                this.cboDevType.DisplayMember = "TypeName";
    
                this.cboBaudRate.DataSource = new CANDevice().BoudRateList;
                this.cboBaudRate.ValueMember = "RateValue";
                this.cboBaudRate.DisplayMember = "RateName";
    
                this.cboDevType.SelectedIndex = 0;
                this.cboDevIndex.SelectedIndex = 0;
                this.cboCANIndex.SelectedIndex = 0;
                this.cboBaudRate.SelectedIndex = 0;
                this.cboFilter.SelectedIndex = 1;
                this.cboFilterMode.SelectedIndex = 0;
                this.cboFrameFormat.SelectedIndex = 0;
                this.cboFrameType.SelectedIndex = 0;
                this.cboSendType.SelectedIndex = 2;//自发自收
                this.cboMode.SelectedIndex = 0;
    
                receiverTimer.Start();
            }
    
            private void btnConnect_Click(object sender, EventArgs e)
            {
                try
                {
                    if (isOpen)//CAN设备已打开
                    {
                        if (objCANDevice != null && objControlCAN != null)
                        {
                            objControlCAN.CloseCAN();
                            btnConnect.Text = "连接";
                            btnStartCAN.Text = "启动CAN";
                            isOpen = false;
                        }
                    }
                    else//CAN设备未打开
                    {
                        objCANDevice = new CANDevice(Convert.ToUInt32(this.cboDevType.SelectedValue), Convert.ToUInt32(this.cboDevIndex.Text), Convert.ToUInt32(this.cboBaudRate.SelectedValue), 2000, Convert.ToUInt32(this.cboCANIndex.Text));
                      
                        objControlCAN = new ControlCAN(objCANDevice);
                        
                        if (objControlCAN.OpenCAN())
                        {
                            btnConnect.Text = "关闭连接";
                            isOpen = true;
                        }
                        else
                            MessageBox.Show("设备打开失败!");
                    }
                }
                catch (Exception ex)
                {
                    MessageBox.Show(ex.Message);
                }
    
            }
    
            private void btnStartCAN_Click(object sender, EventArgs e)
            {
                try
                {
                    if (isOpen)
                    {
                        if (objControlCAN.StartCan())
                        {
                            //初始化CAN参数,可以没有,有默认值
                            _VCI_INIT_CONFIG config = new _VCI_INIT_CONFIG()
                            {
                                AccCode = Convert.ToUInt32("0x" + this.txtAccCode.Text, 16),
                                AccMask = Convert.ToUInt32("0x" + this.txtAccMask.Text, 16),
                                Filter = (Byte)this.cboFilter.SelectedIndex,
                                Mode = (Byte)this.cboFilterMode.SelectedIndex,
                                Timing0 = System.Convert.ToByte("0x" + this.txtTime0.Text, 16),
                                Timing1 = System.Convert.ToByte("0x" + this.txtTime1.Text, 16)
                            };
                            objControlCAN.InitCAN(config);
    
                            this.btnStartCAN.Text = "打开成功";
                        }
                        else
                        {
                            MessageBox.Show("打开失败!");
                            return;
                        }
                    }
                    else
                        MessageBox.Show("请先连接CAN设备!");
                }
                catch (Exception ex)
                {
                    MessageBox.Show(ex.Message);
                }
    
            }
    
            private void btnSend_Click(object sender, EventArgs e)
            {
                _VCI_CAN_OBJ frame = new _VCI_CAN_OBJ();
                //ControlCAN.VCI_CAN_OBJ frame = new ControlCAN.VCI_CAN_OBJ();
                try
                {
                    frame.SendType = (byte)this.cboSendType.SelectedIndex;
                    frame.RemoteFlag = (byte)this.cboFrameFormat.SelectedIndex;
                    frame.ExternFlag = (byte)this.cboFrameType.SelectedIndex;
                    frame.ID = Convert.ToUInt32("0x" + txtID.Text, 16);
    
                    string[] dataStr = this.txtData.Text.Trim().Split(' ');
                    frame.Data = new byte[dataStr.Length];
                    frame.DataLen = (byte)dataStr.Length;
    
                    for (int i = 0; i < dataStr.Length; i++)
                    {
                        frame.Data[i] = Convert.ToByte("0x" + dataStr[i], 16);
                    }
    
                    if (objControlCAN.Transmit(frame))
                    {
                        lbInfo.Items.Add($"发送 {DateTime.Now.ToShortTimeString()} 帧ID:{txtID.Text} 数据:{this.txtData.Text}");
                    }
                    else
                    {
                        MessageBox.Show("发送失败!");
                        return;
                    }
                }
                catch (Exception ex)
                {
                    MessageBox.Show(ex.Message);
                }
            }
    
            private void receiverTimer_Tick(object sender, EventArgs e)
            {
                if (objControlCAN != null && objControlCAN.HasReceive())
                {
                    List<_VCI_CAN_OBJ> objOBJ = objControlCAN._Receive();
                    foreach (var obj in objOBJ)
                    {
                        string str = "接收 ";
                        str += "  帧ID:0x" + System.Convert.ToString((Int32)obj.ID, 16);
                        str += "  帧格式:";
    
                        if (obj.RemoteFlag == 0)
                        {
                            str += "数据: ";
                            foreach (var item in obj.Data)
                            {
                                str += " " + System.Convert.ToString(item, 16);
                            }
                        }
                        lbInfo.Items.Add(str);
                        lbInfo.SelectedIndex = lbInfo.Items.Count - 1;
                    }
                }
            }
        }
    }
    View Code

    有兴趣的盆友可以来看看:github地址

    空间受限不能把项目全部上传,我仅将比较重要的CANDevice、ControlCAN,地址:点击,关于周立功的一些资料比如接口函数定义、相关库函数等,自己可以去官网找,留言我也可以发你~感谢!多多指教~

  • 相关阅读:
    记 · 佛系青年 · 嗯我百度了下
    【CSS】452- 浏览器解析 CSS 样式的过程
    记 · 平安夜 · 平安平安平安
    【Vuejs】451- Vue CLI 首屏优化技巧
    【网站发布】前端自习课
    【CSS】450- 温故而知我不懂的CSS
    【CSS】449- CSS第四级选择器
    RabbitDemo —— Topic
    RabbitDemo —— Fanout
    RabbitDemo —— Direct
  • 原文地址:https://www.cnblogs.com/EasonDongH/p/8721837.html
Copyright © 2020-2023  润新知