欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

自己写C#用汽车诊断卡对MIT电机做调试

程序员文章站 2022-03-15 19:19:20
...

国内现在做MIT电机的工厂很多,有台州海泰(HT),深圳智擎(Dogi),江西老虎(T-MOTOR),江苏伺泰(SteadyWin),
自MIT去年下半年开源了猎豹机器人以及对应的mini-Cheetah actuator(MIT电机)以后,国内有不少做BLDC电机的厂商都开始做这玩意儿了,不过这些企业都是面向无人机市场或者高校研发之类的,
国内做工业伺服的工厂比如汇川伺服,就没有折腾这个

然后我买了个电机来研究一下……

我有两台MIT电机,一台是智擎的
自己写C#用汽车诊断卡对MIT电机做调试
智擎的会附带发你一个专门的驱动板,可以用这个跟电机做速度控制通信,但是,感觉对开发者而言没用,我无法直接用CAN发送报文。
自己写C#用汽车诊断卡对MIT电机做调试

另一个是海泰HT-03规格的MIT电机,同时也买了他们的USB2CAN模块
自己写C#用汽车诊断卡对MIT电机做调试

海泰客服发了我资料,我可以用USB2CAN模块对应的上位机来驱动MIT电机,但他们自己的上位机代码有问题无法实现速度控制只能位置控制,然后,他们拒绝给对应的上位机源码,所以只好自己用其他方式来驱动这个MIT电机。

最终用这个CAN卡来成功驱动了:
自己写C#用汽车诊断卡对MIT电机做调试
店家发CAN卡的时候,还附赠了一对汽车仪表还有发动机接口的线,看来这玩意儿主要是用于汽车维修行业做检测用的。

自己写C#用汽车诊断卡对MIT电机做调试
这是在店家发过来的资料里,自带的C#示例代码的基础上做的
用这种方式来控制电机,比另一位CSDN网友用STM32开发板来控制,我觉得会方便很多,毕竟嵌入式开发有点难,C#简单点……
自己写C#用汽车诊断卡对MIT电机做调试
CSDN惯例贴代码:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;
using System.Runtime.InteropServices;

/*------------兼容ZLG的数据类型---------------------------------*/

//1.ZLGCAN系列接口卡信息的数据类型。
public struct VCI_BOARD_INFO 
{ 
	public UInt16 hw_Version;
    public UInt16 fw_Version;
    public UInt16 dr_Version;
    public UInt16 in_Version;
    public UInt16 irq_Num;
    public byte   can_Num;
    [MarshalAs(UnmanagedType.ByValArray, SizeConst=20)] public byte []str_Serial_Num;
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 40)]
    public byte[] str_hw_Type;
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 4)]
    public byte[] Reserved;
}

/
//2.定义CAN信息帧的数据类型。
unsafe public struct VCI_CAN_OBJ  //使用不安全代码
{
    public uint ID;
    public uint TimeStamp;        //时间标识
    public byte TimeFlag;         //是否使用时间标识
    public byte SendType;         //发送标志。保留,未用
    public byte RemoteFlag;       //是否是远程帧
    public byte ExternFlag;       //是否是扩展帧
    public byte DataLen;          //数据长度
    public fixed byte Data[8];    //数据
    public fixed byte Reserved[3];//保留位

}

//3.定义初始化CAN的数据类型
public struct VCI_INIT_CONFIG 
{
    public UInt32 AccCode;
    public UInt32 AccMask;
    public UInt32 Reserved;
    public byte Filter;   //0或1接收所有帧。2标准帧滤波,3是扩展帧滤波。
    public byte Timing0;  //波特率参数,具体配置,请查看二次开发库函数说明书。
    public byte Timing1;
    public byte Mode;     //模式,0表示正常模式,1表示只听模式,2自测模式
}

/*------------其他数据结构描述---------------------------------*/
//4.USB-CAN总线适配器板卡信息的数据类型1,该类型为VCI_FindUsbDevice函数的返回参数。
public struct VCI_BOARD_INFO1
{
    public UInt16 hw_Version;
    public UInt16 fw_Version;
    public UInt16 dr_Version;
    public UInt16 in_Version;
    public UInt16 irq_Num;
    public byte can_Num;
    public byte Reserved;
    [MarshalAs(UnmanagedType.ByValArray, SizeConst=8)] public byte []str_Serial_Num;
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 16)]
    public byte[] str_hw_Type;
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 16)]
    public byte[] str_Usb_Serial;
}

/*------------数据结构描述完成---------------------------------*/

public struct CHGDESIPANDPORT 
{
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 10)]
    public byte[] szpwd;
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 20)]
    public byte[] szdesip;
    public Int32 desport;

    public void Init()
    {
        szpwd = new byte[10];
        szdesip = new byte[20];
    }
}


namespace WindowsApplication1
{
    public partial class Form1 : Form
    {
        const int DEV_USBCAN = 3;
        const int DEV_USBCAN2 = 4;
         /// <summary>
        /// 
        /// </summary>
        /// <param name="DeviceType"></param>
        /// <param name="DeviceInd"></param>
        /// <param name="Reserved"></param>
        /// <returns></returns>
        /*------------兼容ZLG的函数描述---------------------------------*/
        [DllImport("controlcan.dll")]
        static extern UInt32 VCI_OpenDevice(UInt32 DeviceType, UInt32 DeviceInd, UInt32 Reserved);
        [DllImport("controlcan.dll")]
        static extern UInt32 VCI_CloseDevice(UInt32 DeviceType, UInt32 DeviceInd);
        [DllImport("controlcan.dll")]
        static extern UInt32 VCI_InitCAN(UInt32 DeviceType, UInt32 DeviceInd, UInt32 CANInd, ref VCI_INIT_CONFIG pInitConfig);

        [DllImport("controlcan.dll")]
        static extern UInt32 VCI_ReadBoardInfo(UInt32 DeviceType, UInt32 DeviceInd, ref VCI_BOARD_INFO pInfo);

        [DllImport("controlcan.dll")]
        static extern UInt32 VCI_GetReceiveNum(UInt32 DeviceType, UInt32 DeviceInd, UInt32 CANInd);
        [DllImport("controlcan.dll")]
        static extern UInt32 VCI_ClearBuffer(UInt32 DeviceType, UInt32 DeviceInd, UInt32 CANInd);

        [DllImport("controlcan.dll")]
        static extern UInt32 VCI_StartCAN(UInt32 DeviceType, UInt32 DeviceInd, UInt32 CANInd);
        [DllImport("controlcan.dll")]
        static extern UInt32 VCI_ResetCAN(UInt32 DeviceType, UInt32 DeviceInd, UInt32 CANInd);

        [DllImport("controlcan.dll")]
        static extern UInt32 VCI_Transmit(UInt32 DeviceType, UInt32 DeviceInd, UInt32 CANInd, ref VCI_CAN_OBJ pSend, UInt32 Len);

        [DllImport("controlcan.dll")]
        static extern UInt32 VCI_Receive(UInt32 DeviceType, UInt32 DeviceInd, UInt32 CANInd, ref VCI_CAN_OBJ pReceive, UInt32 Len, Int32 WaitTime);
        
        /*------------其他函数描述---------------------------------*/

        [DllImport("controlcan.dll")]
        static extern UInt32 VCI_ConnectDevice(UInt32 DevType,UInt32 DevIndex);
        [DllImport("controlcan.dll")]
        static extern UInt32 VCI_UsbDeviceReset(UInt32 DevType,UInt32 DevIndex,UInt32 Reserved);
        [DllImport("controlcan.dll")]
        static extern UInt32 VCI_FindUsbDevice(ref VCI_BOARD_INFO1 pInfo);
        /*------------函数描述结束---------------------------------*/

        static UInt32 m_devtype = 4;//USBCAN2

        UInt32 m_bOpen = 0;
        UInt32 m_devind = 0;
        UInt32 m_canind = 0;

        VCI_CAN_OBJ[] m_recobj = new VCI_CAN_OBJ[1000];

        UInt32[] m_arrdevtype = new UInt32[20];

        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            comboBox_DevIndex.SelectedIndex = 0;
            comboBox_CANIndex.SelectedIndex = 0;
            textBox_AccCode.Text = "00000000";
            textBox_AccMask.Text = "FFFFFFFF";
            textBox_Time0.Text = "00";
            textBox_Time1.Text = "14";
            comboBox_Filter.SelectedIndex = 0;              //接收所有类型
            comboBox_Mode.SelectedIndex = 0;                //还回测试模式
            comboBox_FrameFormat.SelectedIndex = 0;
            comboBox_FrameType.SelectedIndex = 0;
            textBox_ID.Text = "1";
            textBox_Data.Text = "FF FF FF FF FF FF FF FC ";

            //
            Int32 curindex = 0;
            comboBox_devtype.Items.Clear();

            curindex = comboBox_devtype.Items.Add("DEV_USBCAN");
            m_arrdevtype[curindex] =  DEV_USBCAN;
            //comboBox_devtype.Items[2] = "VCI_USBCAN1";
            //m_arrdevtype[2]=  VCI_USBCAN1 ;

            curindex = comboBox_devtype.Items.Add("DEV_USBCAN2");
            m_arrdevtype[curindex] = DEV_USBCAN2 ;
            //comboBox_devtype.Items[3] = "VCI_USBCAN2";
            //m_arrdevtype[3]=  VCI_USBCAN2 ;

             comboBox_devtype.SelectedIndex = 1;
            comboBox_devtype.MaxDropDownItems = comboBox_devtype.Items.Count;

        }

        private void Form1_FormClosed(object sender, FormClosedEventArgs e)
        {
            if (m_bOpen==1)
            {
                VCI_CloseDevice(m_devtype, m_devind);
            }
        }

        private void buttonConnect_Click(object sender, EventArgs e)
        {
            if (m_bOpen==1)
            {
                VCI_CloseDevice(m_devtype, m_devind);
                m_bOpen = 0;
            }
            else
            {
                m_devtype = m_arrdevtype[comboBox_devtype.SelectedIndex];

                m_devind=(UInt32)comboBox_DevIndex.SelectedIndex;
                m_canind = (UInt32)comboBox_CANIndex.SelectedIndex;
                if (VCI_OpenDevice(m_devtype, m_devind, 0) == 0)
                {
                    MessageBox.Show("打开设备失败,请检查设备类型和设备索引号是否正确", "错误",
                            MessageBoxButtons.OK, MessageBoxIcon.Exclamation);
                    return;
                }

                m_bOpen = 1;
                VCI_INIT_CONFIG config=new VCI_INIT_CONFIG();
                config.AccCode=System.Convert.ToUInt32("0x" + textBox_AccCode.Text,16);
                config.AccMask = System.Convert.ToUInt32("0x" + textBox_AccMask.Text, 16);
                config.Timing0 = 0x00;// System.Convert.ToByte("0x" + textBox_Time0.Text, 16);
                config.Timing1 = 0x14;// System.Convert.ToByte("0x" + textBox_Time1.Text, 16);
                config.Filter = (Byte)(comboBox_Filter.SelectedIndex+1);
                config.Mode = (Byte)comboBox_Mode.SelectedIndex;
                VCI_InitCAN(m_devtype, m_devind, m_canind, ref config);
            }
            buttonConnect.Text = m_bOpen==1?"断开":"连接";
            timer_rec.Enabled = m_bOpen==1?true:false;
        }

        unsafe private void timer_rec_Tick(object sender, EventArgs e)
        {
            UInt32 res = new UInt32();

            res = VCI_Receive(m_devtype, m_devind, m_canind, ref m_recobj[0],1000, 100);

            /
            //IntPtr[] ptArray = new IntPtr[1];
            //ptArray[0] = Marshal.AllocHGlobal(Marshal.SizeOf(typeof(VCI_CAN_OBJ)) * 50);
            //IntPtr pt = Marshal.AllocHGlobal(Marshal.SizeOf(typeof(IntPtr)) * 1);

            //Marshal.Copy(ptArray, 0, pt, 1);


            //res = VCI_Receive(m_devtype, m_devind, m_canind, pt, 50/*50*/, 100);
            
            if (res == 0xFFFFFFFF) res = 0;//当设备未初始化时,返回0xFFFFFFFF,不进行列表显示。
            String str = "";
            for (UInt32 i = 0; i < res; i++)
            {
                //VCI_CAN_OBJ obj = (VCI_CAN_OBJ)Marshal.PtrToStructure((IntPtr)((UInt32)pt + i * Marshal.SizeOf(typeof(VCI_CAN_OBJ))), typeof(VCI_CAN_OBJ));

                str = "接收到数据: ";
                str += "  帧ID:0x" + System.Convert.ToString(m_recobj[i].ID, 16);
                str += "  帧格式:";
                if (m_recobj[i].RemoteFlag == 0)
                    str += "数据帧 ";
                else
                    str += "远程帧 ";
                if (m_recobj[i].ExternFlag == 0)
                    str += "标准帧 ";
                else
                    str += "扩展帧 ";

                //
                if (m_recobj[i].RemoteFlag == 0)
                {
                    str += "数据: ";
                    byte len = (byte)(m_recobj[i].DataLen % 9);
                    byte j = 0;
                    fixed (VCI_CAN_OBJ* m_recobj1 = &m_recobj[i])
                    {
                        if (j++ < len)
                            str += " " + System.Convert.ToString(m_recobj1->Data[0], 16);
                        if (j++ < len)
                            str += " " + System.Convert.ToString(m_recobj1->Data[1], 16);
                        if (j++ < len)
                            str += " " + System.Convert.ToString(m_recobj1->Data[2], 16);
                        if (j++ < len)
                            str += " " + System.Convert.ToString(m_recobj1->Data[3], 16);
                        if (j++ < len)
                            str += " " + System.Convert.ToString(m_recobj1->Data[4], 16);
                        if (j++ < len)
                            str += " " + System.Convert.ToString(m_recobj1->Data[5], 16);
                        if (j++ < len)
                            str += " " + System.Convert.ToString(m_recobj1->Data[6], 16);
                        if (j++ < len)
                            str += " " + System.Convert.ToString(m_recobj1->Data[7], 16);
                    }
                }

                listBox_Info.Items.Add(str);
                listBox_Info.SelectedIndex = listBox_Info.Items.Count - 1;
            }
            //Marshal.FreeHGlobal(ptArray[0]);
            //Marshal.FreeHGlobal(pt);
        }

        private void button_StartCAN_Click(object sender, EventArgs e)
        {
            if (m_bOpen == 0)
                return;
            VCI_StartCAN(m_devtype, m_devind, m_canind);
        }

        private void button_StopCAN_Click(object sender, EventArgs e)
        {
            if (m_bOpen == 0)
                return;
            VCI_ResetCAN(m_devtype, m_devind, m_canind);
        }

        unsafe private void button_Send_Click(object sender, EventArgs e)
        {
            if(m_bOpen==0)
                return;

            VCI_CAN_OBJ sendobj = new VCI_CAN_OBJ();
            //sendobj.Init();
            sendobj.RemoteFlag = (byte)comboBox_FrameFormat.SelectedIndex;
            sendobj.ExternFlag = (byte)comboBox_FrameType.SelectedIndex;
            sendobj.ID = System.Convert.ToUInt32("0x"+textBox_ID.Text,16);
            int len = (textBox_Data.Text.Length+1) / 3;
            sendobj.DataLen =System.Convert.ToByte(len);
            String strdata = textBox_Data.Text;
            int i=-1;
            if(i++<len-1)
                sendobj.Data[0]=System.Convert.ToByte("0x" +strdata.Substring(i * 3, 2),16);
            if (i++ < len - 1)
                sendobj.Data[1]=System.Convert.ToByte("0x" +strdata.Substring(i * 3, 2),16);
            if (i++ < len - 1)
                sendobj.Data[2]=System.Convert.ToByte("0x" +strdata.Substring(i * 3, 2),16);
            if (i++ < len - 1)
                sendobj.Data[3]=System.Convert.ToByte("0x" +strdata.Substring(i * 3, 2),16);
            if (i++ < len - 1)
                sendobj.Data[4]=System.Convert.ToByte("0x" +strdata.Substring(i * 3, 2),16);
            if (i++ < len - 1)
                sendobj.Data[5]=System.Convert.ToByte("0x" +strdata.Substring(i * 3, 2),16);
            if (i++ < len - 1)
                sendobj.Data[6]=System.Convert.ToByte("0x" +strdata.Substring(i * 3, 2),16);
            if (i++ < len - 1)
                sendobj.Data[7] = System.Convert.ToByte("0x" + strdata.Substring(i * 3, 2), 16);

            if(VCI_Transmit(m_devtype,m_devind,m_canind,ref sendobj,1)==0)
            {
                MessageBox.Show("发送失败", "错误",
                        MessageBoxButtons.OK, MessageBoxIcon.Exclamation);
            }
        }

        private void button_Clear_Click(object sender, EventArgs e)
        {
            listBox_Info.Items.Clear();
        }

        unsafe private void button3_Click(object sender, EventArgs e)
        {
            if (m_bOpen == 0)
                return;

            VCI_CAN_OBJ sendobj = new VCI_CAN_OBJ();
            //sendobj.Init();
            sendobj.RemoteFlag = (byte)comboBox_FrameFormat.SelectedIndex;
            sendobj.ExternFlag = (byte)comboBox_FrameType.SelectedIndex;
            sendobj.ID = System.Convert.ToUInt32("0x" + textBox_ID.Text, 16);
            int len = (textBox_Data.Text.Length + 1) / 3;
            sendobj.DataLen = System.Convert.ToByte(len);
            String strdata = textBox_Data.Text;
            int i = -1;
            if (i++ < len - 1)
                sendobj.Data[0] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[1] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[2] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[3] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[4] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[5] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[6] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[7] = 0xfc;

            if (VCI_Transmit(m_devtype, m_devind, m_canind, ref sendobj, 1) == 0)
            {
                MessageBox.Show("发送失败", "错误",
                        MessageBoxButtons.OK, MessageBoxIcon.Exclamation);
            }
        }

        unsafe private void button4_Click(object sender, EventArgs e)
        {
            if (m_bOpen == 0)
                return;

            VCI_CAN_OBJ sendobj = new VCI_CAN_OBJ();
            //sendobj.Init();
            sendobj.RemoteFlag = (byte)comboBox_FrameFormat.SelectedIndex;
            sendobj.ExternFlag = (byte)comboBox_FrameType.SelectedIndex;
            sendobj.ID = System.Convert.ToUInt32("0x" + textBox_ID.Text, 16);
            int len = (textBox_Data.Text.Length + 1) / 3;
            sendobj.DataLen = System.Convert.ToByte(len);
            String strdata = textBox_Data.Text;
            int i = -1;
            if (i++ < len - 1)
                sendobj.Data[0] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[1] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[2] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[3] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[4] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[5] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[6] = 0xff;
            if (i++ < len - 1)
                sendobj.Data[7] = 0xfd;

            if (VCI_Transmit(m_devtype, m_devind, m_canind, ref sendobj, 1) == 0)
            {
                MessageBox.Show("发送失败", "错误",
                        MessageBoxButtons.OK, MessageBoxIcon.Exclamation);
            }
        }
    }
}

代码里注意, CAN的传输速度必须设置为 1000K。只要速度设对了就可以通信上

使用这个汽车诊断卡,是可以接两路CAN信号的。所以虽然这两个电机的slave id 都是1,但我可以用不同的两个CAN通道来控制,两对CAN线。
自己写C#用汽车诊断卡对MIT电机做调试


记录一下自己踩过的坑:

ARDUINO 坑!
一开始参考的youtube上的瑞士网友skyentific的视频,使用arduin+can shield来做的:
自己写C#用汽车诊断卡对MIT电机做调试
我参照他的方式,买了几张 arduino can shield (因为买错了 can shield) ,都没有成功:
自己写C#用汽车诊断卡对MIT电机做调试
这张卡买来后,一不小心漏焊了SPI的排母,然后排母丢了,就一直都没用上,然后自己费了好大劲把SPI口给焊上:
自己写C#用汽车诊断卡对MIT电机做调试
结果还是不能用!

这是国内改造版本的 arduino seeedio studio 的 CAN shield v1.2 ,非官方原版,它的电子元件跟官方的不一样,我以为是我的焊接问题或国产版电子元件擅自改动的原因,所以又去淘宝买了其他的can shield
自己写C#用汽车诊断卡对MIT电机做调试
这个是skyentific网友在视频里用的sparkfun牌子的can-shield v2.0,拿到手焊接好调试了一个晚上还是没成功,最后发现这板子里的joystick针脚里的left up竟然是短接的,怒,找店家争论,店家说自己的板子绝对没问题,还说让我邮寄退回去,算了先放放,万一我自己把它修好了呢,毕竟只是短接而已……

自己写C#用汽车诊断卡对MIT电机做调试
然后又买了这块板子,依旧是 seeedstudio 的 can-shield v1.2 , 本来就是焊好针脚的,不需要我再焊(我焊接水平不好,另外我只有简单便宜的尖焊头没有刀焊头,所以这个排针对我而言还真不好焊)

这一次使用的时候,发现依然无法跟MIT电机通信,然后终于在seeedstdio官网找到了原因: can-shield v1.2 不支持 1000K的传输速度,

然后又查到了用sparkfun can shield v2.0 不能用的原因,当然有可能是板子本身哪里短接了,但另一方面是国内的大多数 arduino uno , 都是国内简化过的, 擅自把原来的ATMEGA16改成了CH340芯片
CH340芯片属于低速串口芯片,所以我调试不通。
CH340的arduino uno 才16块钱一块,而意大利原版的 ATMEGA16 的 arduino 要100多 , 哎, 心酸……

我放弃了用 arduino 驱动 MIT 电机的这种方式, 鬼知道还有其他哪些坑在这条道路上

2 USB串口调试坑
因为我并不是主职电子领域的,所以对串口不是很懂……
串口的什么 TTL SPI 232 485 协议区别我不太懂,我只知道硬件接线上的区别

智擎的电机,他们的串口我到现在都还没有接通,哎……
反正我调通了海泰电机的串口,就对智擎的放弃了。
他们的串口,搞成了一个USB口,然后发货的时候也没有附带USB线,我向智擎店家要USB口,他们竟然不给,甚至都不卖!我真服了! 然后我自己在淘宝上找了半天也找不到对应的USB口是啥样的,
于是我拆开了他们家的电机,打算自己焊线上去接,
自己写C#用汽车诊断卡对MIT电机做调试
看了一眼,放弃,就我这焊接水平,原指望着能找到一个很大的焊接区域去接线。
但这驱动板上,主控STM32的针脚跟旁边的串口针脚这么近,我要是不小心把主控针脚给连焊了那就完了……

调海泰电机串口时,我先用的 RS232 线来接, 呵呵,当然不行,
店家告诉我这个串口要用 USB2TTL,
然后就去买 USB2TTL,
自己写C#用汽车诊断卡对MIT电机做调试
这两个,便宜的那个是不能用的,因为CH340的速度太慢,无法支持. CP2102 的才行。

最终用 CP2102 的 USB2TTL 调试,才成功。 电机一上电,就会发 mini cheetah 的报文过来, 感动!
毕竟折腾我这么久

3 USB2CAN 的坑
淘宝上就有卖USB2CAN的模块,
自己写C#用汽车诊断卡对MIT电机做调试
这玩意儿是不行的,因为用的CH340芯片,最高速度只能5K,但是MIT电机的需要1000K

后来我买了这个, 还没到货,等到了之后再试试

自己写C#用汽车诊断卡对MIT电机做调试

相关标签: mit 嵌入式