Linux内核配置与构建原理

Kconfig文件

Kconfig是Linux内核中用于配置功能的脚本语言系统,由众多内核源码树中每个目录下的Kconfig文件组成。它定义Linux相关的配置选项层次结构和依赖关系。

menuconfig工具,会抓取Kconfig中的信息,为用户输出友好的交互式菜单选项配置界面。用户在此界面选择需要编译的模块(如Y/N/M),配置结果会保存在.config文件中。

驱动开发:添加新驱动时需在对应目录创建Kconfig条目,并修改上级目录的Kconfig和Makefile以包含新配置。

menuconfig 工具

menuconfig 是 Linux 内核配置的核心工具之一,是基于Kconfig生成的交互式配置工具,提供用户友好的配置菜单界面,简化了内核编译和模块选择的过程。以下是其关键信息:

  1. 基本定义与功能

    Menuconfig 是 make menuconfig 的缩写,基于 ncurses 库实现文本菜单界面。

    用户可通过层级菜单选择或取消内核功能、驱动、文件系统等配置项,无需直接编辑复杂的 .config 文件。

    相较于命令行交互式配置(如 make config)需要逐项回答提问,Menuconfig 提供了更直观的导航和批量操作能力,降低了配置难度。

  2. 核心用途

    内核功能定制:启用/禁用特定功能(如网络协议、硬件驱动、调试工具)。

    模块管理:选择将功能编译为内置模块(Y)、动态加载模块(M)或完全排除(N),优化内核体积。

    硬件适配:为不同硬件平台(如 ARM、X86)配置对应的驱动和优化选项。

    参数调整:设置内核运行参数(如网络栈缓存大小、文件系统行为)。

总结来说,Menuconfig 通过交互式菜单和智能导航设计,将复杂的内核配置转化为可视化的操作流程,是 Linux 系统开发和优化的必备工具。

.config文件

.config文件是配置结果的存储文件,位于内核根目录的.config是用户配置的最终产物,以键值对形式记录所有选项的状态。

.config文件中的配置项被用来: 指导编译系统(make)决定哪些代码需要编译进内核、作为模块或排除。 若不存在,make会使用默认配置(如arch/arm/configs/xxx_defconfig)生成初始文件。

注意事项:直接修改.config可能导致依赖冲突,推荐通过menuconfig调整配置。

Makefile文件

make命令

make命令是编译系统的入口,根据.config和Makefile执行构建操作。make过程会、或者可以做到:

  • 解析.config中的配置项,结合各目录的Makefile决定编译哪些文件。

  • 通过条件编译语句(如obj-$(CONFIG_XXX) += file.o)控制代码的编译方式(内核内置、模块或忽略)。

  • 支持多种编译目标(如make zImage生成内核镜像,make modules编译模块)。

Makefile文件

Makefile 是 自动化构建脚本,定义了软件项目的编译规则、依赖关系和执行顺序。通过 make 命令调用,它能够自动完成代码编译、链接、安装等任务,是 Linux 和嵌入式开发的核心构建工具。

Makefile 的核心作用

  1. 自动化编译 根据源文件(.c.h)的修改时间自动重新编译依赖的文件,避免重复劳动。

  2. 跨平台与交叉编译支持 通过定义变量(如 CCCFLAGS)适配不同编译器(GCC、ARM GCC)和架构(x86、ARM)。

  3. 依赖管理 明确文件间的依赖关系(如 main.o 依赖 main.cutils.h),确保正确编译顺序。

  4. 简化复杂构建流程 将多步骤构建(如清理、安装、生成配置文件)封装为简单命令(如 make cleanmake install)。

  5. 集成其他工具 调用 ldobjcopystrip 等工具生成可执行文件、库文件或烧录镜像。

四者的协作流程

配置阶段:用户通过make menuconfig启动界面,基于Kconfig文件生成菜单树,调整后保存到.config。

编译阶段:make读取.config,根据Makefile中的规则和条件语句编译对应代码。 依赖闭环:Kconfig中的依赖关系确保.config的合法性,而make通过Makefile将配置转化为编译行为

比喻的描述其关系:

Kconfig:定义配置逻辑的“设计图”。-厨师提供的菜品单。 menuconfig:用户交互的“操作界面”。-点餐员。 .config:存储用户选择的“配置文件”。-点餐员根据客人选择的菜品记录下来的点餐单。 Makefile:定义了软件项目的编译规则、依赖关系和执行顺序。-当次做菜的方法和过程。(1.做哪些菜品?读取.config里的配置项,动态调整编译规则。2.每个菜品应该如何烹饪的食材、方法和先后顺序) make:执行编译的“构建引擎”。-给厨师下命令做菜,并输出客户点的菜品。

其他细节


细节1:内核 Makefile 如何动态调整编译规则(基于 .config

一、核心机制:kbuild 系统与配置融合

1. 配置转换为宏定义

当执行 make defconfigmake menuconfig 生成 .config 后,内核会通过脚本(如 scripts/kconfig/confdefconfig)自动生成 autoconf.h 文件。该文件将 .config 中的配置项转换为 C 语言宏定义:

#define CONFIG_GPIO_SUPPORT 1  // 如果配置为 y
#define CONFIG_GPIO_INTERRUPTS m // 如果配置为 m
  • y:表示功能被编译到内核镜像中(直接链接)。

  • m:表示功能被编译为可加载模块(.ko 文件)。

  • n:功能被禁用,不参与编译。

2. Makefile 中的条件编译

内核的 Makefile(尤其是顶层 Makefile 和各子目录的 Makefile)通过以下方式动态调整编译规则:

  • 基于配置启用/禁用源文件

    # 如果 CONFIG_GPIO_SUPPORT 为 y,则将 gpio.o 编译到内核中
    obj-y += gpio.o
    # 如果 CONFIG_GPIO_INTERRUPTS 为 m,则将 gpio_interrupts.o 编译为模块
    obj-m += gpio_interrupts.o

  • 通过 $(CONFIG_XXX) 变量引用配置状态

    ifeq ($(CONFIG_GPIO_SUPPORT), y)
        CFLAGS += -DENABLE_GPIO
    endif
二、内核 Makefile 的动态规则生成
1. obj-$(CONFIG_XXX) 语法

内核 Makefile 使用 obj-$(CONFIG_XXX) 的语法动态控制目标文件的编译方式:

  • obj-y:将文件编译到内核镜像中(当 CONFIG_XXX=y 时生效)。

  • obj-m:将文件编译为模块(当 CONFIG_XXX=m 时生效)。

  • obj-n:明确禁止编译(即使配置为 y 也不编译)。

示例drivers/gpio/Makefile):

obj-y += gpio_core.o   # 总是被编译到内核中,当CONFIG_XXX=y时。
obj-m += gpio_module.o # 仅在 CONFIG_GPIO_SUPPORT=m 时编译为模块
obj-n += deprecated.o  # 显示地禁用旧代码
2. 依赖关系的传递

如果某个配置项依赖于其他配置(如 CONFIG_USB_CORE=yCONFIG_USB_HUB=y 的前提),内核的 Kconfig 会通过 depends on 规则强制关联。对应的 Makefile 会自动忽略无效配置(例如未启用 USB 核心的情况下无法编译 USB HUB)。

三、.config 如何影响编译流程
  1. 生成 autoconf.h

    内核构建时会执行以下步骤:

    make -C /path/to/kernel M=$PWD

    其中,scripts/kconfig/ 目录下的脚本会扫描 .config 并生成 autoconf.h,该文件会被包含到内核源码中(通过 #include <linux/autoconf.h>),从而在 C 代码中可用。

  2. 动态链接对象文件

    • 对于 obj-y 的文件:Makefile 会将这些目标文件直接链接到内核映像(vmlinux)。

    • 对于 obj-m 的文件:Makefile 会将这些文件打包为模块(.ko),并在 modules_install 阶段安装到 /lib/modules/$(KERNEL_VERSION)/kernel/ 目录下。

  3. 条件编译与裁剪

    • 如果 CONFIG_GPIO_SUPPORT=n,内核会跳过所有依赖 GPIO 的代码和模块。

    • 通过 $(CONFIG_XXX) 宏定义,C 代码可以直接判断功能是否启用:

      #ifdef CONFIG_GPIO_SUPPORT
          // 启用 GPIO 功能的代码
      #endif
四、交互场景示例:启用 GPIO 中断支持
1. 配置阶段
make menuconfig   # 打开配置界面
# 导航到 Device Drivers → GPIO Support → 启用 GPIO_INTERRUPTS=m

此时,.config 中新增:

CONFIG_GPIO_SUPPORT=y
CONFIG_GPIO_INTERRUPTS=m
2. 生成配置头文件

运行 makemake prepare,内核会自动生成 autoconf.h,其中包含:

#define CONFIG_GPIO_SUPPORT 1
#define CONFIG_GPIO_INTERRUPTS 1  // 因为 m 被视为 "enabled for module"
3. 动态调整 Makefile
  • drivers/gpio/Makefile 中:

    obj-m += gpio_interrupts.o  # 因为 CONFIG_GPIO_INTERRUPTS=m 有效
  • 如果 CONFIG_GPIO_SUPPORT=n,则 obj-m += gpio_interrupts.o 会被忽略。

4. 编译结果
  • 内核镜像:包含 gpio_core.o(因为 obj-y)。

  • 模块文件:生成 gpio_interrupts.ko(因为 obj-m)。

五、关键实现细节
  1. kbuild 的核心语法 内核 Makefile 使用特殊的 ​kbuild​ 语法,例如:

    ccflags-y:为目标文件添加编译器选项。

    ccflags-y += -I$(PWD)/include

    ldflags-y:为目标文件添加链接器选项。

    ldflags-y += -T $(PWD)/ linker_script.ld
  2. 配置冲突处理 如果 .config 中存在矛盾配置(例如同时设置 CONFIG_USB=yCONFIG_USB=n),内核的 make 命令会报错并终止构建。

  3. 交叉编译适配 在交叉编译环境中,.config 中需显式指定架构和交叉工具链:

    make ARCH=arm CROSS_COMPILE=arm-linux-gnueabi- defconfig
六、总结

内核的 Makefile 通过与 .config 的深度集成,实现了以下功能:

  1. 按需编译:仅启用必要的功能和硬件支持,减少代码量和内存占用。

  2. 模块化支持:通过 obj-m 动态管理可加载模块的编译。

  3. 跨平台兼容:结合交叉编译工具链和架构特定的配置规则。

理解这一机制对嵌入式开发者至关重要,它直接关系到内核定制的灵活性和最终二进制包的优化效果。

细节2:内核构建脚本的核心解析

内核的动态编译规则调整依赖于一系列脚本和工具链的协作,以下是与 .config 配置和 Makefile 生成密切相关的关键脚本及其作用:


一、配置管理与转换脚本
1. confdefconfig
  • 作用:生成默认配置文件(.config)或合并新旧配置。

  • 来源scripts/kconfig/confdefconfig

  • 关键逻辑:

    • defconfigarch/xxx/defconfig 生成初始配置。

    • 通过 olddefconfig 工具将当前 .config 与新内核默认配置对比,保留用户自定义选项。

    • 处理配置冲突(如 yn 同时存在时报错)。

2. kconfig 解析工具
  • 作用:解析 Kconfig 文件并生成配置依赖关系图。

  • 来源scripts/kconfig/parser.c(内核内置的 C 程序)。

  • 输出:

    • 生成 .config 的依赖关系(用于 make menuconfig 的自动折叠菜单)。

    • 生成 symbol_definessymbol_values(辅助配置头文件生成)。


二、配置头文件生成脚本
1. genconfig
  • 作用:将 .config 转换为 autoconf.hversion.h

  • 来源scripts/kconfig/genconfig

  • 关键逻辑:

    • 遍历 .config 中的每个配置项,生成对应的宏定义(如 #define CONFIG_GPIO_SUPPORT 1)。

    • 处理三态配置(m 会生成 CONFIG_GPIO_INTERRUPTS=1,但表示模块化)。

2. check-headers
  • 作用:验证生成的 autoconf.h 是否与内核源码兼容。

  • 来源scripts/kconfig/check-headers

  • 关键逻辑:

    • 检查头文件中是否存在重复定义或冲突的宏。

    • 确保所有依赖项已正确启用(如缺少 CONFIG_USB_CORE 时报错)。


三、Makefile 生成与动态规则处理
1. kbuild 核心脚本
  • 作用:处理 Makefile 的通用规则和依赖关系。

  • 来源scripts/kbuild/Makefile

  • 关键逻辑:

    • 自动包含子目录的 Makefile(通过 include $(SUBDIRS))。

    • 处理 obj-y/obj-m/obj-n 规则,生成编译目标列表。

    • 根据 $(CC)$(CFLAGS) 自动设置编译器和参数。

2. modules.mk
  • 作用:管理内核模块的编译和安装规则。

  • 来源scripts/kbuild/modules.mk

  • 关键逻辑:

    • 定义模块安装路径(/lib/modules/$(KERNEL_VERSION)/kernel/)。

    • 生成模块依赖文件(.modinfo)和符号表(.symvers)。

3. .dependauto-deps
  • 作用:自动生成源文件的依赖关系(类似 GCC 的 -MMD)。

  • 来源scripts/kbuild/dependscripts/kbuild/auto-deps

  • 关键逻辑:

    • 通过 makedepend 工具扫描源文件中的头文件引用。

    • 生成 .d 文件(如 main.o.d),并在 Makefile 中通过 -include $(DEPS) 引入。


四、交叉编译支持脚本
1. cross-compile-check.sh
  • 作用:验证交叉编译环境是否合法。

  • 来源scripts/cross-compile-check.sh

  • 关键逻辑:

    • 检查是否存在 $(CC)$(LD) 变量。

    • 确保交叉工具链支持目标架构(如 armaarch64)。

2. fixup-cross-compile
  • 作用:修复交叉编译时的路径和符号问题。

  • 来源scripts/fixup-cross-compile

  • 关键逻辑:

    • 修改编译器路径以匹配交叉工具链(如 arm-linux-gnueabi-gcc)。

    • 设置 sysroot 和头文件搜索路径(如 --sysroot=/path/to/arm-toolchain)。


五、配置冲突检测与修复
1. check-configuration
  • 作用:检测 .config 中的逻辑矛盾。

  • 来源scripts/kconfig/check-configuration

  • 关键逻辑:

    • 验证 depends onselect 关系的合法性。

    • 检查三态配置是否与布尔配置冲突(如 tristate 配置不能为 n 如果存在依赖项)。

2. silentoldconfig
  • 作用:静默合并新旧配置差异。

  • 来源scripts/kconfig/silentoldconfig

  • 关键逻辑:

    • 将新内核的默认配置与用户旧配置逐项对比。

    • 仅提示用户修改冲突项,其余项自动继承默认值。


六、实战调试脚本
1. make dconfig
  • 作用:基于 .config 生成交互式配置界面。

  • 来源scripts/kconfig/dconfig

  • 关键逻辑:

    • 读取 autoconf.hKconfig 生成动态菜单。

    • 支持在线搜索和配置回滚。

2. make traceconfig
  • 作用:跟踪配置项的依赖关系。

  • 来源scripts/kconfig/traceconfig

  • 关键逻辑:

    • 生成配置项的依赖树(如 CONFIG_GPIO_SUPPORT → CONFIG_ARM)。

    • 输出所有被激活的配置项及其路径。


七、总结:脚本协作流程
  1. 配置阶段

    • 用户通过 make menuconfig 修改 Kconfig,生成 .config

    • confdefconfigsilentoldconfig 处理配置冲突和默认值合并。

  2. 预处理阶段

    • genconfig 生成 autoconf.h,将配置转换为宏定义。

    • kbuild 脚本解析 obj-$(CONFIG_XXX) 规则,生成动态编译目标。

  3. 构建阶段

    • depend 自动生成源文件依赖关系。

    • modules.mk 处理模块编译和安装。

    • 交叉编译脚本(如 cross-compile-check.sh)确保工具链合法。

  4. 验证阶段

    • check-configurationcheck-headers 检测配置合法性。

    • make dconfigmake traceconfig 提供调试支持。


关键脚本与内核构建的关联图

make menuconfig → Kconfig 解析 → .config 生成  
                      ↓  
make defconfig → confdefconfig → 默认配置合并  
                      ↓  
make prepare → genconfig → autoconf.h 生成  
                      ↓  
make all → kbuild/Makefile → obj-$(CONFIG_XXX) 规则应用  
                      ↓  
make modules → modules.mk → 模块编译与安装  
                      ↓  
make clean → depend 清理 .d 文件  

通过以上脚本的协作,内核能够实现 配置驱动开发(Configuration-Driven Development),极大简化了嵌入式设备的定制化过程。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

计算机毕业设计SpringBoot+Vue.js在线课程管理系统(源码+文档+PPT+讲解)

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…

【爬虫基础】第二部分 爬虫基础理论 P3/3

上节内容回顾&#xff1a;【爬虫基础】第一部分 网络通讯 P1/3-CSDN博客 【爬虫基础】第一部分 网络通讯-Socket套接字 P2/3-CSDN博客 【爬虫基础】第一部分 网络通讯-编程 P3/3-CSDN博客 【爬虫基础】第二部分 爬虫基础理论 P1/3-CSDN博客 【爬虫基础】第二部分 爬虫基础理论…

【子网掩码计算器:Python + Tkinter 实现】

子网掩码计算器&#xff1a;Python Tkinter 实现 引言代码功能概述代码实现思路1. 界面设计2. 功能实现3. 事件处理 子网掩码计算器实现步骤1. 导入必要的库2. 定义主窗口类 SubnetCalculatorApp3. 创建菜单栏4. 创建界面组件5. 判断 IP 地址类别6. 计算子网信息7. 其他功能函…

【第十节】C++设计模式(结构型模式)-Flyweight( 享元)模式

目录 一、问题背景 二、模式选择 三、代码实现 四、总结讨论 一、问题背景 享元模式&#xff08;Flyweight Pattern&#xff09;在对象存储优化中的应用 在面向对象系统的设计与实现中&#xff0c;创建对象是最常见的操作之一。然而&#xff0c;如果一个应用程序使用了过多…

macOS - 使用 tmux

文章目录 安装 tmux使用更多快捷键说明 安装 tmux brew install tmux使用 在终端输入 tmux 进入 tmux 界面&#xff0c;然后 输入 Control Option B 进入交互模式 输入 % 左右分栏&#xff0c;" 上下分割 上一个窗格&#xff1a;{&#xff0c;下一个&#xff1a;} PS…

【洛谷贪心算法题】P1094纪念品分组

该题运用贪心算法&#xff0c;核心思想是在每次分组时&#xff0c;尽可能让价格较小和较大的纪念品组合在一起&#xff0c;以达到最少分组的目的。 【算法思路】 输入处理&#xff1a;首先读取纪念品的数量n和价格上限w&#xff0c;然后依次读取每件纪念品的价格&#xff0c;…

16. LangChain实战项目2——易速鲜花内部问答系统

需求简介 易束鲜花企业内部知识库如下&#xff1a; 本实战项目设计一个内部问答系统&#xff0c;基于这些内部知识&#xff0c;回答内部员工的提问。 在前面课程的基础上&#xff0c;需要安装的依赖包如下&#xff1a; pip install docx2txt pip install qdrant-client pip i…

Minio搭建并在SpringBoot中使用完成用户头像的上传

Minio使用搭建并上传用户头像到服务器操作,学习笔记 Minio介绍 minio官网 MinIO是一个开源的分布式对象存储服务器&#xff0c;支持S3协议并且可以在多节点上实现数据的高可用和容错。它采用Go语言开发&#xff0c;拥有轻量级、高性能、易部署等特点&#xff0c;并且可以自由…

Spring AI:让AI应用开发更简单

文章目录 引言什么是Spring AI&#xff1f;核心特性 Spring AI的核心组件ChatClient&#xff1a;聊天模型示例代码图示 ImageClient&#xff1a;图像生成示例代码图示 Prompt Templates&#xff1a;提示词模板示例代码 Spring AI的优势示例项目&#xff1a;智能机票助手代码实现…

【C】链式二叉树算法题1 -- 单值二叉树

leetcode链接https://leetcode.cn/problems/univalued-binary-tree/description/ 1 题目描述 如果二叉树每个节点都具有相同的值&#xff0c;那么该二叉树就是单值二叉树。只有给定的树是单值二叉树时&#xff0c;才返回 true&#xff1b;否则返回 false。 示例 1&#xff1…

什么是最终一致性,它对后端系统的意义是什么

最终一致性(Eventual Consistency)是分布式系统中的一种一致性模型。与传统的强一致性模型不同,最终一致性并不要求系统在任何时刻都保持一致,而是保证在足够的时间后,所有节点的数据最终会达到一致的状态。换句话说,系统允许短时间内出现数据的不一致性,但最终会通过某…

掌握大模型高效任务流搭建(一):构建LangChain任务流

前言&#xff1a; 在LangChain框架中&#xff0c;“链”占据着核心地位。它允许我们将众多任务模块串联起来&#xff0c;构建出富有弹性的任务流。借助这种链式结构&#xff0c;我们能够处理复杂的逻辑&#xff0c;并实现任务的自动化。在实际场景里&#xff0c;链式操作极大地…

目标检测——数据处理

1. Mosaic 数据增强 Mosaic 数据增强步骤: (1). 选择四个图像&#xff1a; 从数据集中随机选择四张图像。这四张图像是用来组合成一个新图像的基础。 (2) 确定拼接位置&#xff1a; 设计一个新的画布(输入size的2倍)&#xff0c;在指定范围内找出一个随机点&#xff08;如…

塑造网络安全的关键事件

注&#xff1a;本文为 “网络安全” 相关文章合辑。 机翻&#xff0c;未校。 Timeline of Cyber Security: Key Events that Shaped the Field 网络安全时间表&#xff1a;塑造该领域的关键事件 October 29, 2023 Cyberattacks are an everyday threat, always changing. T…

题解 | 牛客周赛82 Java ABCDEF

目录 题目地址 做题情况 A 题 B 题 C 题 D 题 E 题 F 题 牛客竞赛主页 题目地址 牛客竞赛_ACM/NOI/CSP/CCPC/ICPC算法编程高难度练习赛_牛客竞赛OJ 做题情况 A 题 判断字符串第一个字符和第三个字符是否相等 import java.io.*; import java.math.*; import java.u…

Redis 高可用性:如何让你的缓存一直在线,稳定运行?

&#x1f3af; 引言&#xff1a;Redis的高可用性为啥这么重要&#xff1f; 在现代高可用系统中&#xff0c;Redis 是一款不可或缺的分布式缓存与数据库系统。无论是提升访问速度&#xff0c;还是实现数据的高效持久化&#xff0c;Redis 都能轻松搞定。可是&#xff0c;当你把 …

uniapp-原生android插件开发摘要

uni-app在App侧的原生扩展插件&#xff0c;支持使用java、object-c等原生语言编写&#xff0c;从HBuilderX 3.6起&#xff0c;新增支持了使用uts来开发原生插件。 基础项目 UniPlugin-Hello-AS工程请在App离线SDK中查找 基础项目(App离线SDK)已经配置好了自定义插件所需要的…

【定昌Linux系统】部署了java程序,设置开启启动

将代码上传到相应的目录&#xff0c;并且配置了一个.sh的启动脚本文件 文件内容&#xff1a; #!/bin/bash# 指定JAR文件的路径&#xff08;如果JAR文件在当前目录&#xff0c;可以直接使用文件名&#xff09; JAR_FILE"/usr/local/java/xs_luruan_client/lib/xs_luruan_…

SpringBoot源码解析(十):应用上下文AnnotationConfigServletWebServerApplicationContext构造方法

SpringBoot源码系列文章 SpringBoot源码解析(一)&#xff1a;SpringApplication构造方法 SpringBoot源码解析(二)&#xff1a;引导上下文DefaultBootstrapContext SpringBoot源码解析(三)&#xff1a;启动开始阶段 SpringBoot源码解析(四)&#xff1a;解析应用参数args Sp…

Unity小功能实现:鼠标点击移动物体

1、功能描述 当玩家点击鼠标时&#xff0c;场景中的物体会移动到鼠标点击的位置。这个功能可以用于控制角色移动、放置物体等场景。 2、实现步骤 创建Unity项目&#xff1a;首先&#xff0c;打开Unity并创建一个新的3D项目。 添加3D物体&#xff1a;在场景中创建一个3D物体&am…