ubuntu 使用虚拟can 与 socketCAN使用

2023-05-16

原文链接:https://blog.csdn.net/xiandang8023/article/details/127990159

创建虚拟CAN接口

在Linux上能使用虚拟CAN接口之前,需要在终端执行以下三个步骤:

加载vcan内核模块: sudo modprobe vcan

创建虚拟CAN接口: sudo ip link add dev vcan0 type vcan

将虚拟CAN接口处于在线状态: sudo ip link set up vcan0 或 sudo ip link set dev vcan0 up

将虚拟CAN接口处于离线状态: sudo ip link set down vcan0 或 sudo ip link set dev vcan0 down

删除虚拟CAN接口: sudo ip link del dev vcan0

然后,通过命令ip addr | grep "can" 来验证是否可用并处于在线状态

也可以通过shell脚本来自动化实现以上步骤:

创建一个vcan.sh文件。然后将其标记为可执行文件: chmod +x ~/vcan.sh,之后执行这个文件 ./vcan.sh

#!/bin/bash
# Make sure the script runs with super user priviliges.
[ "$UID" -eq 0 ] || exec sudo bash "$0" "$@"
# Load the kernel module.
modprobe vcan
# Create the virtual CAN interface.
ip link add dev vcan0 type vcan
# Bring the virutal CAN interface online.
ip link set up vcan0

使用 can-utils 测试CAN通信

接下来我们要基于上面创建的虚拟CAN接口,来测试一下CAN通信情况。工具包 can-utils 是一个命令行工具,可以完美的满足我们的需求。我们只需要在电脑上安装一下这个工具包即可:

Ubuntu/Debian: sudo apt install can-utils

接下来,我们打开两个终端窗口,一个是用来查看所有的CAN消息,另一个是用来发送CAN消息。

在用来查看CAN消息的终端中执行以下命令:

candump -tz vcan0

在用来发送CAN消息的终端中,模拟发送CAN请求:

cansend vcan0 123#00FFAA5501020304

在发送完CAN请求后,就会在第一个查看CAN消息的终端中看到发送的CAN消息:

发送CAN信息

melvyn@melvyn:~$ cansend vcan0 123#00FFAA5501020304
melvyn@melvyn:~$ cansend vcan0 123#00FFAA5501020304

查看CAN信息

 (000.000000)  vcan0  123   [8]  00 FF AA 55 01 02 03 04
 (001.919922)  vcan0  123   [8]  00 FF AA 55 01 02 03 04

从上面的输出可以验证使用新创建的vcan0虚拟CAN接口可以正常进行CAN通信。

CAN数据读取的代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <linux/can.h>
#include <linux/can/raw.h>
#include <net/if.h>
int main(void)
{
    struct ifreq ifr = {0};
    struct sockaddr_can can_addr = {0};
    struct can_frame frame = {0};
    int sockfd = -1;
    int i;
    int ret;

    /* 打开套接字 */
    sockfd = socket(PF_CAN, SOCK_RAW, CAN_RAW);
    if(0 > sockfd) {
        perror("socket error");
        exit(EXIT_FAILURE);
    }

    /* 指定can0设备 */
    strcpy(ifr.ifr_name, "vcan0");
    ioctl(sockfd, SIOCGIFINDEX, &ifr);
    can_addr.can_family = AF_CAN;
    can_addr.can_ifindex = ifr.ifr_ifindex;

    /* 将can0与套接字进行绑定 */
    ret = bind(sockfd, (struct sockaddr *)&can_addr, sizeof(can_addr));
    if (0 > ret) {
        perror("bind error");
        close(sockfd);
        exit(EXIT_FAILURE);
    }

    /* 设置过滤规则 */
    //setsockopt(sockfd, SOL_CAN_RAW, CAN_RAW_FILTER, NULL, 0);

    /* 接收数据 */
    for ( ; ; ) {
        if (0 > read(sockfd, &frame, sizeof(struct can_frame))) {
            perror("read error");
            break;
        }

        /* 校验是否接收到错误帧 */
        if (frame.can_id & CAN_ERR_FLAG) {
            printf("Error frame!\n");
            break;
        }

        /* 校验帧格式 */
        if (frame.can_id & CAN_EFF_FLAG)    //扩展帧
            printf("扩展帧 <0x%08x> ", frame.can_id & CAN_EFF_MASK);
        else        //标准帧
            printf("标准帧 <0x%03x> ", frame.can_id & CAN_SFF_MASK);

        /* 校验帧类型:数据帧还是远程帧 */
        if (frame.can_id & CAN_RTR_FLAG) {
            printf("remote request\n");
            continue;
        }

        /* 打印数据长度 */
        printf("[%d] ", frame.can_dlc);

        /* 打印数据 */
        for (i = 0; i < frame.can_dlc; i++)
            printf("%02x ", frame.data[i]);
        printf("\n");
    }

    /* 关闭套接字 */
    close(sockfd);
    exit(EXIT_SUCCESS);
}

CAN数据写入的代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <linux/can.h>
#include <linux/can/raw.h>
#include <net/if.h>
int main(void)
{
    struct ifreq ifr = {0};
    struct sockaddr_can can_addr = {0};
    struct can_frame frame = {0};
    int sockfd = -1;
    int ret;

    /* 打开套接字 */
    sockfd = socket(PF_CAN, SOCK_RAW, CAN_RAW);
    if(0 > sockfd) {
        perror("socket error");
        exit(EXIT_FAILURE);
    }

    /* 指定can0设备 */
    strcpy(ifr.ifr_name, "vcan0");
    ioctl(sockfd, SIOCGIFINDEX, &ifr);
    can_addr.can_family = AF_CAN;
    can_addr.can_ifindex = ifr.ifr_ifindex;

    /* 将can0与套接字进行绑定 */
    ret = bind(sockfd, (struct sockaddr *)&can_addr, sizeof(can_addr));
    if (0 > ret) {
        perror("bind error");
        close(sockfd);
        exit(EXIT_FAILURE);
    }

    /* 设置过滤规则:不接受任何报文、仅发送数据 */
    setsockopt(sockfd, SOL_CAN_RAW, CAN_RAW_FILTER, NULL, 0);

    /* 发送数据 */
    frame.data[0] = 0xA0;
    frame.data[1] = 0xB0;
    frame.data[2] = 0xC0;
    frame.data[3] = 0xD0;
    frame.data[4] = 0xE0;
    frame.data[5] = 0xF0;
    frame.can_dlc = 6;    //一次发送6个字节数据
    frame.can_id = 0x123;//帧ID为0x123,标准帧

    for ( ; ; ) {

        ret = write(sockfd, &frame, sizeof(frame)); //发送数据
        if(sizeof(frame) != ret) { //如果ret不等于帧长度,就说明发送失败
            perror("write error");
            goto out;
        }

        sleep(1);        //一秒钟发送一次
    }

out:
    /* 关闭套接字 */
    close(sockfd);
    exit(EXIT_SUCCESS);
}

运行结果

补充 CAN 的常用操作命令:

可以使用ip命令来查看或设置CAN,使用ifconfig或ip命令来开启/关闭CAN,canconfig工具来配置和调试CAN,cansend 和 candump用于收发CAN报文。

#ifconfig -a //查到当前can网络 can0 can1,包括收发包数量、是否有错误等等
#ip link set vcan0 down //关闭can设备;或使用ifconfig canX down
#ip link set vcan0 up //开启can设备;或使用ifconfig canX up
#ip -details link show vcan0 //显示can设备详细信息;
#ip link set vcan0 up type can bitrate 250000 //设置can波特率
#canconfig vcan0 ctrlmode loopback on //回环测试;
#canconfig vcan0 restart // 重启can设备;
#canconfig vcan0 stop //停止can设备;
#canecho vcan0 //查看can设备总线状态;
#candump vcan0 //接收can总线发来的数据;
#cansend vcan0 --identifier=ID+数据 //发送数据;
#candump vcan0 --filter=ID:mask//使用滤波器接收ID匹配的数据

复杂版程序

原文链接:https://zhuanlan.zhihu.com/p/429616854

#ifndef _CAN_H_
#define _CAN_H_

#ifdef __cplusplus
extern "C" {
#endif

int iface_is_up(char *iface);
int can_iface_down(char *iface);
int can_iface_up(char *iface, int baudrate, int fdon, int dbaudrate);
int can_open(char *iface, int canid, int canfd_on);
int can_send(int sock, int canid, unsigned char *data, int len);
int canfd_send(int sock, int canid, unsigned char *data, int len);
int can_recv(int sock, unsigned char *data, int len, int *canid);
int canfd_recv(int sock, unsigned char *data, int len, int *canid);
int can_close(int sock);

#ifdef __cplusplus
}
#endif

#endif
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <unistd.h>
#include <net/if.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <linux/can.h>
#include <linux/can/raw.h>
#include "can.h"

#define CAN_IFACE_TXQUEUELEN 100

int iface_is_up(char *iface)
{
    int sock;
    struct ifreq ifr;
    int ret;
    sock = socket(PF_CAN, SOCK_RAW, CAN_RAW);
    if (sock < 0) {
        perror("socket err.\n");
        return 0;
    }
    strcpy(ifr.ifr_name, iface);
    ioctl(sock, SIOCGIFFLAGS, &ifr);
    if (ifr.ifr_ifru.ifru_flags & IFF_RUNNING) {
        ret = 1;
    } else {
        ret = 0;
    }
    close(sock);
    return ret;
}

int can_iface_down(char *iface)
{
    char cmd[128];
    snprintf(cmd, sizeof(cmd), "sudo ifconfig %s down", iface);
    system(cmd);
    return 0;
}

int can_iface_up(char *iface, int baudrate, int fdon, int dbaudrate)
{
    char cmd[256];
    int len = 0;
    if (iface_is_up(iface)) {
        printf("Can Iface [%s] is up\n", iface);
        return 0;
    }
    printf("Try to Start Can Iface [%s]...\n", iface);
    len += snprintf(cmd + len, sizeof(cmd) - len, "sudo ip link set %s up type can bitrate %d ", iface, baudrate);
    if (fdon) {
        len += snprintf(cmd + len, sizeof(cmd) - len, " fd on dbitrate %d ", dbaudrate);
    } else {
        // len += snprintf(cmd + len, sizeof(cmd) - len, " fd off");
    }
    system(cmd);
    sleep(1);
    len = snprintf(cmd, sizeof(cmd), "sudo ifconfig %s txqueuelen %d", iface, CAN_IFACE_TXQUEUELEN);
    system(cmd);
    printf("Can Iface [%s] start finish.\n", iface);
    return 0;
}

int can_open(char *iface, int canid, int canfd_on)
{
    int sock;
    struct ifreq ifr;
    struct sockaddr_can addr;
    struct can_filter rfilter[1];
    if (!iface_is_up(iface)) {
        perror("iface err..\n");
        return -1;
    }
    sock = socket(PF_CAN, SOCK_RAW, CAN_RAW);
    if (sock < 0) {
        perror("socket err.\n");
        return -1;
    }
    strcpy(ifr.ifr_name, iface);
    ioctl(sock, SIOCGIFINDEX, &ifr);
    addr.can_family = AF_CAN;
    addr.can_ifindex = ifr.ifr_ifindex;
    if (bind(sock, (struct sockaddr *)&addr, sizeof(addr))) {
        perror("bind err.\n");
        return -1;
    }
    if (canid > 0) {
        rfilter[0].can_id = canid;
        rfilter[0].can_mask = CAN_SFF_MASK;
        if (setsockopt(sock, SOL_CAN_RAW, CAN_RAW_FILTER, &rfilter, sizeof(rfilter))) {
            perror("setsockopt [SOL_CAN_RAW/CAN_RAW_FILTER] err.\n");
            return -1;
        }
    }
    if (canfd_on > 0) {
        if (setsockopt(sock, SOL_CAN_RAW, CAN_RAW_FD_FRAMES, &canfd_on, sizeof(canfd_on))) {
            perror("setsockopt [SOL_CAN_RAW/CAN_RAW_FD_FRAMES] err.\n");
            return -1;
        }
    }
    /* int loopback = 0;
    if (setsockopt(sock, SOL_CAN_RAW, CAN_RAW_LOOPBACK, &loopback, sizeof(loopback))) {
                perror("setsockopt [SOL_CAN_RAW/CAN_RAW_LOOPBACK] err.\n");
                return -1;
        } */
    return sock;
}

int can_send(int sock, int canid, unsigned char *data, int len)
{
    int nr;
    struct can_frame frame;
    if (len > sizeof(frame.data)) {
        printf("can_send: len more than 8 is not support.\n");
        return -1;
    }
    frame.can_dlc = len;
    frame.can_id = canid;
    memcpy(frame.data, data, len);
    nr = write(sock, &frame, sizeof(struct can_frame));
    if (nr != sizeof(struct can_frame)) {
        printf("can_send: write ret=%d[%s].\n", nr, strerror(errno));
        return -1;
    }
    return frame.can_id;
}

int can_recv(int sock, unsigned char *data, int len, int *canid)
{
    int nr;
    struct can_frame frame;
    nr = read(sock, &frame, sizeof(struct can_frame));
    if (nr < 0) {
        // perror("can_recv: read err.\n");
        return -1;
    }
    if (nr == 0) {
        perror("can_recv: read ret 0, peer may closed.\n");
        return -1;
    }
    if (canid) {
        *canid = frame.can_id;
    }
    if (len < frame.can_dlc) {
        printf("can_recv: recv data buffer too small, data has trunced.\n");
        memcpy(data, frame.data, len);
        return len;
    } else {
        memcpy(data, frame.data, frame.can_dlc);
    }
    return frame.can_dlc;
}

int canfd_send(int sock, int canid, unsigned char *data, int len)
{
    int nr;
    struct canfd_frame frame;
    if (len > sizeof(frame.data)) {
        printf("can_send: len more than 8 is not support.\n");
        return -1;
    }
    frame.len = len;
    frame.can_id = canid;
    frame.flags = 0xF;
    memcpy(frame.data, data, len);
    nr = write(sock, &frame, sizeof(struct canfd_frame));
    if (nr != sizeof(struct canfd_frame)) {
        perror("can_send: write err.\n");
        return -1;
    }
    return frame.can_id;
}

int canfd_recv(int sock, unsigned char *data, int len, int *canid)
{
    int nr;
    struct canfd_frame frame;
    nr = read(sock, &frame, sizeof(struct canfd_frame));
    if (nr < 0) {
        //perror("can_recv: read err.\n");
        return -1;
    }
    if (nr == 0) {
        perror("can_recv: read ret 0, peer may closed.\n");
        return -1;
    }
    if (len < frame.len) {
        printf("can_recv: recv data buffer too small, data has trunced.\n");
        memcpy(data, frame.data, len);
        return len;
    } else {
        memcpy(data, frame.data, frame.len);
    }
    if (canid) {
        *canid = frame.can_id;
    }
    return frame.len;
}

int can_close(int sock)
{
    return close(sock);
}
#include "can.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <time.h>
#include <fcntl.h>

unsigned char sent_data[8];

void show_hex(unsigned char * data, int len)
{
    int i = 0;
    while(i < len)
    {
        if(!(i % 16))
        {
            printf("0x%08x ", i);
        }
        printf("%02x ", data[i]);
        if(i % 16 == 15)
        {
            printf("\n");
        }
        i = i + 1;
    }
    printf("\n\n");
}

void * can_aux_loop(int sock_)
{
    int nr, canid, i, failure = 0;
    unsigned char data[8];
    while(true)
    {
        usleep(100);
        nr = can_recv(sock_, data, sizeof(data), &canid);
        if(nr > 0)
        {
            printf("can_recv canid: %03x.\n", canid);
            show_hex(data, nr);
        }
        else
        {
            failure = failure + 1;
            if(failure > 20)
            {
                break;
            }
            continue;
        }
        i = 0;
        while(i < 8)
        {
            if(data[i] != sent_data[i])
            {
                break;
            }
            i = i + 1;
        }
        if(i == 8)
        {
            break;
        }
        else
        {
            failure = failure + 1;
            if(failure > 20)
            {
                break;
            }
        }
    }
}

void * can_send_loop(int sock0, int sock_, int send_canid)
{
    fcntl(sock_, F_SETFL, fcntl(sock_, F_GETFL, 0) | O_NONBLOCK);
    int nr, i;
    time_t t;
    srand((unsigned) time(&t));
    unsigned char send_byte, data[8];
    while(true)
    {
        send_byte = rand() % 256;
        memset(data, send_byte, sizeof(data));
        nr = can_send(sock0, send_canid, data, sizeof(data));
        if(nr < 0)
        {
            continue;
        }
        printf("can_send [8 byte of '%02x'] canid: %03x.\n\n", send_byte, send_canid);
        i = 0;
        while(i < 8)
        {
            sent_data[i] = data[i];
            i = i + 1;
        }
        can_aux_loop(sock_);
    }
    can_close(sock0);
    can_close(sock_);
}

void * can_recv_loop(int sock1)
{
    int nr, canid;
    unsigned char data[8];
    while(true)
    {
        nr = can_recv(sock1, data, sizeof(data), &canid);
        if(nr > 0)
        {
            printf("can_recv canid: %03x.\n", canid);
            show_hex(data, nr);
        }
    }
    can_close(sock1);
}

int main(int argc, char ** argv)
{
    char * can_device = argv[2];
    int send_canid = 0x20C, recv_canid = 0x000, baudrate = 1000000, fd = 0, dbaudrate = 4000000;
    can_iface_up(can_device, baudrate, fd, dbaudrate);
    if(strcmp(argv[1], "send") == 0)
    {
        int sock0 = can_open(can_device, send_canid, fd), sock_ = can_open(can_device, send_canid, fd);
        if(sock0 < 0)
        {
            printf("can_app: open can_device [%s] failed.\n", can_device);
            return -1;
        }
        can_send_loop(sock0, sock_, send_canid);
    }
    else if(strcmp(argv[1], "recv") == 0)
    {
        int sock1 = can_open(can_device, recv_canid, fd);
        if(sock1 < 0)
        {
            printf("can_app: open can_device [%s] failed.\n", can_device);
            return -1;
        }
        can_recv_loop(sock1);
    }
    return 0;
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

ubuntu 使用虚拟can 与 socketCAN使用 的相关文章

随机推荐

  • CUDA Stream流并发性

    目录 1 CUDA 中的异步命令 2 默认流 3 Multistream多流示例 异构计算是指高效地使用系统中的所有处理器 xff0c 包括 CPU 和 GPU 为此 xff0c 应用程序必须在多个处理器上并发执行函数 CUDA 应用程序通
  • Madagascar环境下编程

    引用 原创 Madagascar环境下编程 2013 07 17 04 50 34 转载 标签 xff1a 教育 分类 xff1a madagascar 本文转载自seismig 原创 Madagascar环境下编程 Madagascar是
  • mySQL(关系型数据库管理系统)编辑

    收藏 2906 1034 mySQL xff08 关系型数据库管理系统 xff09 编辑 MySQL 1 是一个 关系型数据库管理系统 xff0c 由瑞典 MySQL AB公司开发 xff0c 目前属于 Oracle公司 MySQL是最流行
  • CPU的核心数、线程数的关系和区别

    我们在选购电脑的时候 xff0c CPU是一个需要考虑到核心因素 xff0c 因为它决定了电脑的性能等级 CPU从早期的单核 xff0c 发展到现在的双核 xff0c 多核 CPU除了核心数之外 xff0c 还有线程数之说 xff0c 下面
  • STM32单片机,下载器下载完程序能正常跑起来,断电再上电程序不运行

    晶振坏了 转载于 https www cnblogs com god of death p 7050281 html
  • CUDA性能优化----warp深度解析

    CUDA性能优化 warp深度解析 2017 01 12 16 41 07 分类 xff1a HPC amp CUDA优化 标签 xff1a gpu cuda hpc 举报 字号 订阅 下载LOFTER 我的照片书 1 引言 CUDA性能优
  • 螺旋桨转矩

    xfeff xfeff 在螺旋桨气动力分析时 xff0c 首先应用翼型理论进行螺旋桨叶素分析 利用翼型升阻特性数据 xff0c 回避了有限机翼的展弦比问题 xff0c 诱导流动由涡流模型确定 xff0c 取决于桨叶数目 间距以及作用于每片桨
  • 给初学者们讲解人工神经网络(ANN)

    1 介绍 这份教学包是针对那些对人工神经网络 xff08 ANN xff09 没有接触过 基本上完全不懂的一批人做的一个简短入门级的介绍 我们首先简要的引入网络模型 xff0c 然后才开始讲解ANN的相关术语 作为一个应用的案例 xff0c
  • OpenMP基本概念

    OpenMP是一种用于共享内存并行系统的多线程程序设计方案 xff0c 支持的编程语言包括C C 43 43 和Fortran OpenMP提供了对并行算法的高层抽象描述 xff0c 特别适合在多核CPU机器上的并行程序设计 编译器根据程序
  • 散度和旋度的物理意义是什么?

    高等数学学的时间有点久远了 xff0c 最近需要推倒一些公式 xff0c 以前高数学的时候这公式那定理的都没说什么物理意义 xff0c 现在接触的问题都是具有一定物理意义的 xff0c 感觉对不上 xff0c 回来找找资料好好理解一下 xf
  • 格林公式、高斯公式及斯托克斯公式的理解及相互关系

    最近要推倒波动方程积分解 xff0c 要对散度 旋度以及他们之间的相互关系有一个理解 看了两天 xff0c 自己认为理解的差不多了 xff0c 现在写在这个地方 xff0c 作为笔记 xff0c 以后忘记了拿过来看一下 xff0c 加深一下
  • Radon变换理论介绍

    本人最近在研究Radon变换 xff0c 在查阅了各种资料之后在此写下个人的理解 xff0c 希望与各位牛牛进行交流共同进步 xff0c 也使得理解更加深刻些 Radon变换的本质是将原来的函数做了一个空间转换 xff0c 即 xff0c
  • test

    lt DOCTYPE html gt lt html lang 61 34 en 34 gt lt head gt lt meta charset 61 34 utf 8 34 gt lt meta http equiv 61 34 X U
  • BIT内存顺序

    机器的最小寻址单位是字节 xff0c bit无法寻址 xff0c 也就没有高低地址和起始地址的概念 xff0c 我们需要定义一下bit的 地址 以一个字节为例 xff0c 我们把从左到右的8个bit的位置 position 命名按顺序命名如
  • 无人驾驶感知篇之融合(五)

    今天早上看到上海新增一万七千左右 xff0c 看的真的很揪心 xff01 希望白衣战士能早点战胜这场疫情 xff0c 期待明天能有好消息 xff01 今天具体讲讲多贝叶斯估计算法的原理 xff0c 多贝叶斯估计法的主要思想是将传感器信息依据
  • MAC地址的介绍(单播、广播、组播、数据收发)

    MAC地址组成 网络设备的MAC地址是全球唯一的 MAC地址长度为48比特 xff0c 通常用十六进制表示 MAC地址包含两部分 xff1a 前24比特是组织唯一标识符 xff08 OUI xff0c OrganizationallyUni
  • stm32通用定时器输出PWM控制舵机

    stm32的通用定时器有TIM2 TIM3 TIM4 TIM5 xff0c 每个定时器都有独立的四个通道可以作为 xff1a 输入捕获 输出比较 PWM输出 单脉冲模式输出等 stm32除了基本定时器 xff0c 其他定时器都能输出PWM
  • Linux内核Socket CAN中文文档

    自己在年假中空闲之余翻译的内核中Socket CAN的文档 xff0c 原文地址在 xff1a http lxr linux no linux 43 v2 6 34 Documentation networking can txt 但是这篇
  • c/c++自定义通讯协议(TCP/UDP)

    前言 xff1a TCP与UDP是大家耳熟能详的两种传输层通信协议 xff0c 本质区别在于传输控制策略不相同 xff1a 使用TCP协议 xff0c 可以保证传输层数据包能够有序地被接受方接收到 xff0c 依赖其内部一系列复杂的机制 x
  • ubuntu 使用虚拟can 与 socketCAN使用

    原文链接 xff1a https blog csdn net xiandang8023 article details 127990159 创建虚拟CAN接口 在Linux上能使用虚拟CAN接口之前 xff0c 需要在终端执行以下三个步骤