iMX6ULL 库移植 | Libgpiod 库的交叉编译及使用指南(linux)

GPIO口的操作,是很常见的功能。传统的GPIO sysfs接口已被弃用。自Linux 4.8起,内核提供了全新的操作gpio的方式libgpiod(C library and tools for interacting with the linux GPIO character device),当然也更高效,推荐使用。

libgpiod简介

libgpiod - 用于与Linux GPIO字符设备进行交互的C库和工具(gpiod代表GPIO设备)

libgpiod/libgpiod.git - C library and tools for interacting with the linux GPIO character device

自Linux 4.8起,GPIO sysfs接口已被弃用。用户空间应改用字符设备。该库封装了ioctl调用和数据结构,提供了一个简单直观的API。

新的字符设备接口保证在关闭设备文件描述符后释放所有分配的资源,并添加了一些在已过时的sysfs接口中不存在的新功能(如事件轮询、一次设置/读取多个值或开源和开漏GPIO)。

不幸的是,不再可以仅使用标准命令行工具与Linux设备文件进行交互。这就是创建一个库的原因,它封装了繁琐的、基于ioctl的内核-用户空间交互,并提供了一组方便的函数和不透明的数据结构。 此外,该项目还包含一组命令行工具,可方便地将用户脚本转换为使用字符设备。

旧的方式 GPIO sysfs 接口使用

在linux4.8之前,没有libgpiod,传统操作GPIO口使用的是sysfs方式控制。使用方式举例如下:

1.sysfs 先导出 GPIO 17 和 GPIO 18

echo 17 > /sys/class/gpio/export
echo 18 > /sys/class/gpio/export

现在再去 ls /sys/class/gpio/ 目录,就会多出来 gpio17 和 gpio18 目录。

2.设置 gpio 模式为输出。

echo out > /sys/class/gpio/gpio17/direction
echo out > /sys/class/gpio/gpio18/direction

3.设置 gpio 高电平点亮 LED,点亮绿色(G)灯。

echo 1 > /sys/class/gpio/gpio18/value
#熄灭写 0 即可
echo 0 > /sys/class/gpio/gpio18/value

如何构建libgpiod

这是一个相当标准的autotools项目。核心C库除了标准C库与GNU扩展之外,没有任何外部依赖项。 命令行工具可选择依赖于libedit以获得交互功能。 要构建项目(包括命令行工具),

运行:

./autogen.sh --enable-tools=yes --prefix=<安装路径>makemake install

autogen脚本将执行./configure并将所有命令行参数传递给它。

有关所有configure功能,请参阅:./configure --help。

自带工具

目前有六个可用的命令行工具:

* gpiodetect - 列出系统上存在的所有gpiochips,它们的名称、标签和GPIO线数

* gpioinfo - 列出线路、它们的gpiochip、偏移量、名称和方向,如果在使用中,则列出使用者名称和任何其他配置的属性,如活动状态、偏置、驱动、边缘检测和去抖动周期

* gpioget - 读取指定GPIO的值

* gpioset - 设置指定GPIO的值,保持线路状态直到进程被终止或退出

* gpiomon - 等待GPIO上的边缘事件,指定要监视的边缘,处理多少个事件后退出,或者是否将事件报告到控制台

* gpionotify - 等待GPIO信息的更改,指定要监视的更改,处理多少个事件后退出,或者是否将事件报告到控制台

工具使用举例

# 检测可用的gpiochips。 
$ gpiodetect 
gpiochip0 [pinctrl-bcm2711]
gpiochip1 [raspberrypi-exp-gpio]# 读取gpiochip上所有的信息。 
$ gpioinfo -c 1 
gpiochip1 - 8个: 0:“BT_ON” 输出 1:“WL_ON” 输出 2:“PWR_LED_OFF” 输出低电平 使用者=“led1” 3:“GLOBAL_RESET” 输出 4:“VDD_SD_IO_SEL” 输出 使用者=“vdd-sd-io” 5:“CAM_GPIO” 输出 使用者=“cam1_regulator” 6:“SD_PWR_ON” 输出 使用者=“sd_vcc_reg” 7:“SD_OC_N” 输入 # 读取特定的信息。 
$ ./gpioinfo PWR_LED_OFF STATUS_LED_G_CLK GLOBAL_RESET 
gpiochip0 42 “STATUS_LED_G_CLK” 输出 使用者=“led0” 
gpiochip1 2 “PWR_LED_OFF” 输出低电平 使用者=“led1” 
gpiochip1 3 “GLOBAL_RESET” 输出 # 按名称读取单个GPIO的值。 
$ gpioget RXD1 
“RXD1” = 激活 # 按芯片和偏移量读取单个GPIO的值。 
$ gpioget -c 0 15 
“15” = 激活 # 以数字值的形式读取单个GPIO的值。 
$ gpioget --numeric RXD1 
1 # 同时读取两个值。将的活动状态设置为低电平,并且不使用带引号的名称。 
$ gpioget --active-low --unquoted GPIO23 GPIO24 
GPIO23 = 激活 GPIO24 = 激活 # 设置的值,并保持该直到被终止。 
$ gpioset GPIO23=1 # 设置两个的值,然后使其成为守护进程并保持。 
$ gpioset --daemonize GPIO23=1 GPIO24=0 # 设置单个的值,保持20毫秒,然后退出。 
$ gpioset --hold-period 20ms -t0 GPIO23=1 # 在GPIO22上以1Hz频率闪烁LED 
$ gpioset -t500ms GPIO22=1 # 在GPIO22上以1Hz频率和20%的工作比闪烁LED 
$ gpioset -t200ms,800ms GPIO22=1 # 以交互方式设置一些(需要--enable-gpioset-interative) 
$ gpioset --interactive --unquoted GPIO23=inactive GPIO24=active 
gpioset> get 
GPIO23 = inactive GPIO24 = active 
gpioset> toggle 
gpioset> get 
GPIO23 = 激活 GPIO24 = inactive 
gpioset> set GPIO24=1 
gpioset> get 
GPIO23 = 激活 GPIO24 = 激活 
gpioset> toggle 
gpioset> get 
GPIO23 = inactive GPIO24 = inactive 
gpioset> toggle GPIO23 
gpioset> get 
GPIO23 = 激活 GPIO24 = inactive 
gpioset> exit # 等待单个GPIO上的三个上升沿事件,然后退出。 
$ gpiomon --num-events=3 --edges=rising GPIO22 
10002.907638045 上升沿 “GPIO22” 
10037.132562259 上升沿 “GPIO22” 
10047.179790748 上升沿 “GPIO22” # 在单个GPIO上等待三个边缘事件,使用本地时间和不带引号的名称,然后退出。 
$ gpiomon --num-events=3 --edges=both --localtime --unquoted GPIO22 
2022-11-15T10:36:59.109615508 上升沿 GPIO22 
2022-11-15T10:36:59.129681898 下降沿 GPIO22 
2022-11-15T10:36:59.698971886 上升沿 GPIO22 # 等待下降沿事件并使用自定义输出格式。 
$ gpiomon --format="%e %c %o %l %S" --edges=falling -c gpiochip0 22 
2 gpiochip0 22 GPIO22 10946.693481859 
2 gpiochip0 22 GPIO22 10947.025347604 
2 gpiochip0 22 GPIO22 10947.283716669 
2 gpiochip0 22 GPIO22 10947.570109430 
... # 阻塞直到发生边缘事件。不打印任何内容。 
$ gpiomon --num-events=1 --quiet GPIO22 # 监视多个,在第一个边缘事件后退出。 
$ gpiomon --quiet --num-events=1 GPIO5 GPIO6 GPIO12 GPIO17 # 监视的信息更改。 
$ gpionotify GPIO23 
11571.816473718 请求 “GPIO23” 
11571.816535124 释放 “GPIO23” 
11572.722894029 请求 “GPIO23” 
11572.722932843 释放 “GPIO23” 
11573.222998598 请求 “GPIO23” 
... # 监视的请求,报告UTC时间和不带引号的名称。 
$ gpionotify --utc --unquoted GPIO23 
2022-11-15T03:05:23.807090687Z 请求 GPIO23 
2022-11-15T03:05:23.807151390Z 释放 GPIO23 
2022-11-15T03:05:24.784984280Z 请求 GPIO23 
2022-11-15T03:05:24.785023873Z 释放 GPIO23 
... # 监视多个,在第一个请求后退出。 
$ gpionotify --quiet --num-events=1 --event=requested GPIO5 GPIO6 GPIO12 GPIO17 # 阻塞直到被释放。 
$ gpionotify --quiet --num-events=1 --event=released GPIO6 

 引脚的计算方法

 如  PG14 == 206 (line)  (6x32+14),PG10 == 202(line)

查看GPIO的占用情况

cat /sys/kernel/debug/gpio

结果报错,提示没有这个文件:cat: can't open '/sys/kernel/debug/gpio': No such file or directory。上网查了下,需要先执行这个命令:

mount -t debugfs debugfs /sys/kernel/debug

libgpiod库的交叉编译

1. 安装交叉编译工具链:根据目标平台的架构和操作系统,下载并安装相应的交叉编译工具链。例如,对于ARM架构的Linux系统,您可以使用arm-linux-gnueabi工具链。

2. 下载libgpiod源代码:从libgpiod的官方仓库或官方网站下载最新的源代码压缩包。

3. 解压源代码:将下载的压缩包解压到您选择的目录中。

4. 进入源代码目录:使用终端进入解压后的libgpiod源代码目录。

5. 设置环境变量:根据您的交叉编译工具链,设置以下环境变量:

   #export CROSS_COMPILE=<交叉编译工具链前缀>export CROSS_COMPILE=/opt/okt507/buildroot/host/bin/aarch64-linux-gnu-export CC=${CROSS_COMPILE}gccexport CXX=${CROSS_COMPILE}g++export AR=${CROSS_COMPILE}arexport AS=${CROSS_COMPILE}asexport LD=${CROSS_COMPILE}ldexport RANLIB=${CROSS_COMPILE}ranlibexport STRIP=${CROSS_COMPILE}strip

单个设置麻烦,可以保存脚本文件,例如 setenv.sh ,然后在终端中运行以下命令来设置环境变量并构建libgpiod库: 

source setenv.sh

这将加载脚本中的环境变量,并使其在当前终端会话中生效。然后,您可以继续执行构建步骤来编译和安装libgpiod库。

遇到错误

yang@ubuntu:~/okt507/gpiod/libgpiod-1.6.4$ ./autogen.sh --enable-tools=yes
./autogen.sh: 17: ./autogen.sh: autoreconf: not founderror: possibly undefined macro: AC_CHECK_HEADERSconfigure.ac:190: the top level
configure.ac:91: error: possibly undefined macro: AC_CHECK_HEADERSIf this token and others are legitimate, please use m4_pattern_allow.See the Autoconf documentation.
configure.ac:183: error: possibly undefined macro: AC_LANG_PUSH
configure.ac:185: error: possibly undefined macro: AC_LANG_POP
autoreconf: error: /usr/bin/autoconf failed with exit status: 1

需要安装依赖:

sudo apt-get install -y autoconf automake libtool
sudo apt-get install autoconf-archive
sudo apt-get install m4sudo apt install pkg-config

其中最后一个 pkg-config一定不能忘了,很重要的依赖,否则容易报以下错误:

configure.ac:91: error: possibly undefined macro: AC_CHECK_HEADERSIf this token and others are legitimate, please use m4_pattern_allow.See the Autoconf documentation.

具体原因,这里有解释:

Why 'Possibly Undefined Macro' Means Install pkg-config | AE1020: Lazy Notebook

 最后,开始交叉编译:

#加载环境变量
source setenv.sh
./autogen.sh --enable-tools=yes --host=arm-linux#--prefix指定make install的安装目录
./configure --enable-tools=yes --host=arm-linux --prefix=/home/yang/okt507/gpiod/build

Ubuntu关闭后台更新解决重启时间过长问题

Ubuntu虚拟机重启发现用时过长,显示:A stop job is running for Unattended Upgrades Shutdown (10s / 30 min)

经过百度,发现是Ubuntu会自动更新,但这时间也太长了,于是果断禁用。

执行命令:

sudo dpkg-reconfigure unattended-upgrades

选择no并按ENTER以禁用无人参与的升级。
环境变量setenv.sh脚本

#!/bin/bash# 设置交叉编译工具链前缀
export CROSS_COMPILE=<交叉编译工具链前缀># 设置交叉编译工具链
export CC=${CROSS_COMPILE}gcc
export CXX=${CROSS_COMPILE}g++
export AR=${CROSS_COMPILE}ar
export AS=${CROSS_COMPILE}as
export LD=${CROSS_COMPILE}ld
export RANLIB=${CROSS_COMPILE}ranlib
export STRIP=${CROSS_COMPILE}strip# 设置安装路径
#export INSTALL_PATH=<安装路径>

6. 配置构建:运行以下命令配置构建过程:

./autogen.sh --enable-tools=yes --prefix=<安装路径>

<安装路径> 替换为您希望安装libgpiod的路径。

7. 运行构建:运行以下命令开始构建libgpiod库:

8. 安装库:运行以下命令将构建好的库安装到指定的安装路径:

c API使用

#include <gpiod.h>
#include <stdio.h>
#include <unistd.h>#ifndef	CONSUMER
#define	CONSUMER	"Consumer"
#endifint blend_led(struct gpiod_line *r, int r_val, struct gpiod_line *g, int g_val, struct gpiod_line *b, int b_val);/**
* GPIO 16 <-> R
* GPIO 17 <-> B
* GPIO 18 <-> G
*/
int main(int argc, char **argv)
{char *chipname = "gpiochip0";unsigned int line_num_16 = 16;	// GPIO 16unsigned int line_num_17 = 17;	// GPIO 17unsigned int line_num_18 = 18;	// GPIO 18struct gpiod_chip *chip;struct gpiod_line *line16, *line17, *line18;int ret;chip = gpiod_chip_open_by_name(chipname);if (!chip) {printf("Open chip by name failed. name: %s\n", chipname);goto end;}line16 = gpiod_chip_get_line(chip, line_num_16);if (!line16) {printf("Get line failed. line_num: %u\n", line_num_16);goto close_chip;}line17 = gpiod_chip_get_line(chip, line_num_17);if (!line17) {printf("Get line failed. line_num: %u\n", line_num_17);goto release_line16;}line18 = gpiod_chip_get_line(chip, line_num_18);if (!line18) {printf("Get line failed. line_num: %u\n", line_num_18);goto release_line17;}ret = gpiod_line_request_output(line16, CONSUMER, 0);if (ret < 0) {printf("Request line16 as output failed\n");goto release_line18;}ret = gpiod_line_request_output(line17, CONSUMER, 0);if (ret < 0) {printf("Request line17 as output failed\n");goto release_line18;}ret = gpiod_line_request_output(line18, CONSUMER, 0);if (ret < 0) {printf("Request line18 as output failed\n");goto release_line18;}//Rret = blend_led(line16, 1, line17, 0, line18, 0);if (ret < 0) {printf("Set output failed.\n");goto release_line18;}sleep(1);//Gret = blend_led(line16, 0, line17, 0, line18, 1);if (ret < 0) {printf("Set output failed.\n");goto release_line18;}sleep(1);//Bret = blend_led(line16, 0, line17, 1, line18, 0);if (ret < 0) {printf("Set output failed.\n");goto release_line18;}sleep(1);//yellowret = blend_led(line16, 1, line17, 0, line18, 1);if (ret < 0) {printf("Set output failed.\n");goto release_line18;}sleep(1);ret = blend_led(line16, 1, line17, 1, line18, 0);if (ret < 0) {printf("Set output failed.\n");goto release_line18;}sleep(1);ret = blend_led(line16, 0, line17, 1, line18, 1);if (ret < 0) {printf("Set output failed.\n");goto release_line18;}sleep(1);ret = blend_led(line16, 1, line17, 1, line18, 1);if (ret < 0) {printf("Set output failed.\n");goto release_line18;}release_line18:gpiod_line_release(line18);
release_line17:gpiod_line_release(line17);
release_line16:gpiod_line_release(line16);
close_chip:gpiod_chip_close(chip);
end:return 0;
}int blend_led(struct gpiod_line *r, int r_val, struct gpiod_line *g, int g_val, struct gpiod_line *b, int b_val){int ret = 0;ret = gpiod_line_set_value(r, r_val);if (ret < 0) {printf("Set r output failed. val: %u\n", r_val);return ret;}ret = gpiod_line_set_value(g, g_val);if (ret < 0) {printf("Set g output failed. val: %u\n", g_val);return ret;}ret = gpiod_line_set_value(b, b_val);if (ret < 0) {printf("Set b output failed. val: %u\n", b_val);return ret;}return 0;
}

其他资源

qlibgpiod/libgpiod.git - C library and tools for interacting with the linux GPIO character device

Libgpiod库的使用,点亮LED_猪突猛进进进的博客-CSDN博客

百度安全验证

autoreconf执行,出现undefined macro问题_linuxarmsummary的博客-CSDN博客

RaspberryPi 4B 使用 libgpiod 操作 gpio_树莓派4b引脚图_TYYJ-洪伟的博客-CSDN博客

Why 'Possibly Undefined Macro' Means Install pkg-config | AE1020: Lazy Notebook

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/112009.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Nuxt 菜鸟入门学习笔记三:视图

文章目录 入口文件组件 Components页面 Pages布局 Layouts Nuxt 官网地址&#xff1a; https://nuxt.com/ Nuxt 提供多个组件层来实现应用程序的用户界面。 入口文件 App.vue组件 Components页面 Pages布局 Layouts 下面逐一进行介绍。 入口文件 默认情况下&#xff0c;Nu…

vue3使用Elementplus 动态显示菜单icon不生效

1.问题描述 菜单icon由后端提供&#xff0c;直接用的字符串返回&#xff0c;前端使用遍历显示&#xff0c;发现icon不会显示 {id: 8, path:/userManagement, authName: "用户管理", icon: User, rights:[view]}, <el-menu-item :index"menu.path" v-f…

常用数据库备份方法,sql数据库备份方法

在信息时代&#xff0c;数据成为了公司的主要资产。然而&#xff0c;数据的安全性和完整性也成为企业管理的重要组成部分。因此&#xff0c;数据库备份至关重要。本文将详细介绍几种常见的数据库备份方法。 全备份 全备份是指数据库中所有数据的备份&#xff0c;包括数据文件、…

五、多表查询-4.6练习

一、准备数据 【效果展示】 emp1表&#xff08;员工表&#xff09;&#xff1a; dept1表&#xff08;部门表&#xff09;&#xff1a; salgrade表&#xff08;薪资等级表&#xff09;&#xff1a; 二、案例 1、查询员工的姓名、年龄、职位、部门信息&#xff08;隐式内连接&am…

SpringBoot + layui 框架实现一周免登陆功能

✅作者简介&#xff1a;2022年博客新星 第八。热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏…

MySQL的日志undolog、binlog、redolog

1. 日志层次 binlog是Server层&#xff0c;undolog和redolog是innodb引擎层特有的。 2. 记录了什么 & 作用 binlog 记录了所有数据库结构变更和表数据修改的SQL日志。 主要用于数据备份和主从复制&#xff0c;比如误删数据了可以用binlog找回。 undolog 如下图&#…

Verilog 实现状态机自动售卖机

Verilog 实现状态机自动售卖机 教学视频&#xff1a;https://www.bilibili.com/video/BV1Ve411x75W?p33&spm_id_frompageDriver&vd_source19ae31dff4056e52d2729a4ca212602b 功能需求 使用1元、2元、5元面值的纸币进行支付&#xff0c;获取6元的物品&#xff0c;不设…

在el-tree懒加载中进行局部刷新

在进行懒加载的树组件中&#xff0c;操作子节点新增、修改以及删除操作时&#xff0c;需要对树组件进行局部刷新&#xff1a; /* 懒加载 */ async loadNode(node, resolve) {if (node.level 0) {// 异步加载根节点数据const data await fn({ parentId: });resolve(data);thi…

linux中学习控制进程的要点

1. 进程创建 1.1 fork函数 #include <unistd.h> pid_t fork(void); 返回值&#xff1a;自进程中返回0&#xff0c;父进程返回子进程id&#xff0c;出错返回-1 进程调用fork&#xff0c;当控制转移到内核中的fork代码后&#xff0c;内核会做以下操作 分配新的内存块和…

19.CSS雨云动画特效

效果 源码 <!DOCTYPE html> <html lang="en"> <head><meta charset="UTF-8"><title>Cloud & Rain Animation</title><link rel="stylesheet" href="style.css"> </head> <bo…

专题:平面、空间直线参数方程下的切线斜率问题

本文研究平面、空间直线在参数方程形式下&#xff0c;切线斜率&#xff08;即导数&#xff09;如何表示的问题。 如上图所示。 设 y f ( x ) &#xff0c; x φ ( t ) &#xff0c; y ψ ( t ) 当 t t 0 时&#xff0c; x x 0 &#xff0c; y y 0 &#xff0c;即点 A 坐…

最简单vue获取当前地区天气--高德开放平台实现

目录 前言 一、注册成为高德平台开发者 二、注册天气key 1.点击首页右上角打开控制台 2.创建新应用 三、vue项目使用 1.打开vue项目找到public下的index.html&#xff0c;如果是vue3的话直接在主目录打开index.html文件就行&#xff0c;主要就是打开出口文件 ​编辑 2.根据高德…

元矿山下的音视频应用

// 近年来&#xff0c;矿业的技术和管理模式随着元宇宙的火爆和自动驾驶技术的发展逐渐变化、升级&#xff0c;进而衍生出元矿山的概念&#xff0c;音视频技术也在其中成为了关键一环。LiveVideoStackCon 2023 上海站邀请了来自希迪智驾的任思亮&#xff0c;为大家分享希迪智…

无限计算力:探索云计算的无限可能性

这里写目录标题 前言云计算介绍服务模型&#xff1a; 应用领域&#xff1a;云计算主要体现在生活中的地方云计算未来发展的方向 前言 云计算是一种基于互联网的计算模型&#xff0c;通过它可以实现资源的共享、存储、管理和处理。它已经成为许多个人、企业和组织的重要技术基础…

MySQL数据库学习【基础篇】

&#x1f4c3;基础篇 下方链接使用科学上网速度可能会更加快一点哦&#xff01; 请点击查看数据库MySQL笔记大全 通用语法及分类 DDL: 数据定义语言&#xff0c;用来定义数据库对象&#xff08;数据库、表、字段&#xff09;DML: 数据操作语言&#xff0c;用来对数据库表中的…

mybatis与spring集成与spring aop集成pagehelper插件

Mybatis与Spring的集成 Mybatis是一款轻量级的ORM框架&#xff0c;而Spring是一个全栈式的框架&#xff0c;二者的结合可以让我们更加高效地进行数据持久化操作。 Mybatis与Spring的集成主要有两种方式&#xff1a;使用Spring的Mybatis支持和使用Mybatis的Spring支持。 使用…

再获殊荣 | 美格智能高算力AI模组SNM970荣获物联网行业“通信技术创新奖”

8月28日&#xff0c;由高科技行业门户OFweek维科网主办的OFweek 2023&#xff08;第八届&#xff09;物联网产业大会暨评选颁奖典礼在深圳福田会展中心隆重举行。会上正式公布了OFweek 2023&#xff08;第八届&#xff09;物联网与人工智能行业年度评选奖项&#xff0c;美格智能…

springmvc没有绿标,怎么配置tomcat插件运行?

一、添加插件后&#xff0c;刷新&#xff0c;自动从maven仓库下载tomcat插件 二、写好项目后&#xff0c;添加tomcat配置 三、即可点击绿标运行

flutter高德地图大头针

1、效果图 2、pub get #地图定位 amap_flutter_map: ^3.0.0 amap_flutter_location: ^3.0.0 3、上代码 import dart:async; import dart:io;import package:amap_flutter_location/amap_flutter_location.dart; import package:amap_flutter_location/amap_location_option…

用变压器实现德-英语言翻译【02/8】: 位置编码

一、说明 本文是“用变压器实现德-英语言翻译”系列的第二篇。它从头开始引入位置编码。然后&#xff0c;它解释了 PyTorch 如何实现位置编码。接下来是变压器实现。 二、技术背景 位置编码用于为序列中的每个标记或单词提供相对位置。阅读句子时&#xff0c;每个单词都依赖于它…