ARM uboot 源码分析5 -启动第二阶段

2023-11-11

一、start_armboot 解析6

1、console_init_f

在这里插入图片描述


(1) console_init_f 是 console(控制台)的第一阶段初始化。_f 表示是第一阶段初始化,_r 表示第二阶段初始化。有时候初始化函数不能一次一起完成,中间必须要夹杂一些代码,因此将完整的一个模块的初始化分成了 2 个阶段。(我们的 uboot 中 start_armboot 的 826 行进行了 console_init_r 的初始化)

在这里插入图片描述

在这里插入图片描述


(2) console_init_f 在 uboot/common/console.c 中,仅仅是对 gd->have_console 设置为 1 而已,其他事情都没做。


2、display_banner

在这里插入图片描述


(1) display_banner 用来串口输出显示 uboot 的 logo。
在**这**里插入图片描述


(2) display_banner 中使用 printf 函数向串口输出了 version_string 这个字符串。那么上面的分析表示, console_init_f 并没有初始化好 console, 怎么就可以 printf 了呢?

(3) 通过追踪 printf 的实现,发现 printf->puts,而 puts 函数中会判断当前 uboot 中 console 有没有被初始化好

如果 console 初始化好了,则调用 fputs 完成串口发送(这条线才是控制台);如果 console 尚未初始化好,则会调用 serial_puts (再调用 serial_putc 直接操作串口寄存器进行内容发送)

(4) 控制台也是通过串口输出,非控制台也是通过串口输出。究竟什么是控制台?和不用控制台的区别?

实际上分析代码会发现,控制台就是一个用软件虚拟出来的设备,这个设备有一套专用的通信函数(发送、接收···),控制台的通信函数,最终会映射到硬件的通信函数中来实现。uboot 中,实际上控制台的通信函数是直接映射到硬件串口的通信函数中的,也就是说 uboot 中用没用控制台,其实并没有本质差别

(5) 但是在别的体系中,控制台的通信函数映射到硬件通信函数时,可以用软件来做一些中间优化,譬如说缓冲机制。(操作系统中的控制台都使用了缓冲机制,所以有时候我们 printf 了内容,但是屏幕上并没有看到输出信息,就是因为被缓冲了。我们输出的信息只是到了 console 的 buffer 中,buffer 还没有被刷新到硬件输出设备上,尤其是在输出设备是LCD屏幕时)

(6) U_BOOT_VERSION 在 uboot 源代码中找不到定义,这个变量实际上是在 makefile 中定义的,然后在编译时生成的 include/version_autogenerated.h 中用一个宏定义来实现的。

在这里插入图片描述


3、print_cpuinfo

在这里插入图片描述

在这里插入图片描述

(1) uboot启动过程中:

CPU:  S5PV210@1000MHz(OK)
        APLL = 1000MHz, HclkMsys = 200MHz, PclkMsys = 100MHz
        MPLL = 667MHz, EPLL = 96MHz
                       HclkDsys = 166MHz, PclkDsys = 83MHz
                       HclkPsys = 133MHz, PclkPsys = 66MHz
                       SCLKA2M  = 200MHz
Serial = CLKUART 

这些信息都是 print_cpuinfo 打印出来的。

(2) 回顾 ARM 裸机中时钟配置一章的内容,比对这里调用的函数中计算各种时钟的方法,自己去慢慢分析体会这些代码的原理和实现方法。


二、start_armboot 解析7

1、checkboard

在这里插入图片描述
在这里插入图片描述

(1) checkboard 看名字是检查、确认开发板的意思。这个函数的作用就是检查当前开发板是哪个开发板并且打印出开发板的名字。


2、init_func_i2c

在这里插入图片描述

(1) 这个函数实际没有被执行,X210 的 uboot 中并没有使用 I2C。如果将来我们的开发板要扩展 I2C 来接外接硬件,则在 x210_sd.h 中配置相应的宏即可开启。


3、uboot 学习实践

(1) 对 uboot 源代码进行完修改(修改内容根据自己的理解和分析来修改)

在这里插入图片描述

(2) make distclean 然后 make x210_sd_config 然后 make。

make distclean
make x210_sd_config
make -j8

在这里插入图片描述

(3) 编译完成得到 u-boot.bin,然后去烧录。烧录方法按照 x210 刷机教程 讲的 linux 下使用 dd 命令来烧写的方法来烧写。

(4) 在 uboot_jiuding 文件夹下,就有 sd_fusing 烧录所需的文件。

在这里插入图片描述


注意,由于这个路径的文件是被九鼎官方编译过的,里面的文件是 64 bit 格式的;而我们的 linux 系统是 32 bit ,所以必须先make clean 清除后,重新编译生成文件。

在这里插入图片描述

烧写过程:
第一步:进入 sd_fusing 目录下
第二步:make clean
第三步:make

在这里插入图片描述

第四步:插入 SD 卡,ls /dev/sd* 得到 SD 卡在 ubuntu 中的设备号(一般是/dev/sdb,注意 SD 卡要连接到虚拟机 ubuntu 中,不要接到 windows 中)

在这里插入图片描述

注意,我们的 sd_fusing.sh 文件就是写的 /dev/sdb ,如有需要可以自行更改。

在这里插入图片描述


第五步:修改 sd_fusing.sh 文件:u-boot.bin.
在这里插入图片描述

./sd_fusing.sh /dev/sdb 完成烧录(注意不是 sd_fusing2.sh)。

在这里插入图片描述

(5) 总结:uboot 就是个庞大点复杂点的裸机程序而已,我们完全可以对他进行调试。调试的方法就是按照上面步骤,根据自己对代码的分析和理解对代码进行更改,然后重新编译烧录运行,根据运行结果来学习。


修改效果成功

在这里插入图片描述


三、start_armboot 解析8

1、dram_init

在这里插入图片描述
在这里插入图片描述

(1) dram_init 看名字是关于 DDR 的初始化。疑问:在汇编阶段已经初始化过 DDR 了,否则也无法 relocate 到这里的第二部分运行,怎么在这里又初始化 DDR ?

(2) dram_init 都是在给 gd->bd 里面关于 DDR 配置部分的全局变量赋值,让 gd->bd 数据记录下当前开发板的 DDR 的配置信息,以便uboot中使用内存。

(3) 从代码来看,其实就是初始化 gd->bd->bi_dram 这个结构体数组。


2、display_dram_config

在这里插入图片描述
在这里插入图片描述

(1) 看名字意思就是打印显示 dram 的配置信息。

(2) 启动信息中的:(DRAM: 512 MB) 就是在这个函数中打印出来的。


uboot bdinfo 命令

思考:如何在 uboot 运行中得知 uboot 的 DDR 配置信息?uboot 中有一个命令叫 bdinfo ,这个命令可以打印出 gd->bd 中记录的所有硬件相关的全局变量的值,因此可以得知 DDR 的配置信息。

DRAM bank   = 0x00000000
-> start    = 0x30000000
-> size     = 0x10000000
DRAM bank   = 0x00000001
-> start    = 0x40000000
-> size     = 0x10000000

在这里插入图片描述


3、init_sequence总结

(1) 都是板级硬件的初始化以及 gd、gd->bd 中的数据结构的初始化。譬如:
网卡初始化、机器码(gd->bd->bi_arch_number)、内核传参 DDR 地址(gd->bd->bi_boot_params)、Timer4 初始化为 10ms 一次、波特率设置(gd->bd->bi_baudrate和gd->baudrate)、console 第一阶段初始化(gd->have_console 设置为 1 )、打印uboot 的启动信息、打印 cpu 相关设置信息、检查并打印当前开发板名字、DDR 配置信息初始化(gd->bd->bi_dram)、打印 DDR 总容量。


四、start_armboot 解析9

1、CFG_NO_FLASH

在这里插入图片描述

在这里插入图片描述

(1) 虽然 NandFlashNorFlash 都是 Flash,但是一般 NandFlash 会简称为 Nand 而不是 Flash,一般讲 Flash 都是指的 Norflash。这里的 2 行代码是 Norflash 相关的

(2) flash_init 执行的是开发板中对应的 NorFlash 的初始化、display_flash_config 打印的也是 NorFlash 的配置信息(Flash: 8 MB 就是这里打印出来的)。但是实际上 X210 中是没有 Norflash 的。所以这两行代码是可以去掉的(我也不知道为什么没去掉?猜测原因有可能是去掉这两行代码,会导致别的地方工作不正常,需要花时间去移植调试,然后移植的人就懒得弄。实际上不去掉,除了显示有 8MB Flash 实际没用之外,也没有别的影响)

CONFIG_VFD 和 CONFIG_LCD 是显示相关的,这个是 uboot 中自带的 LCD 显示的软件架构。但是实际上我们用 LCD 而没有使用 uboot 中设置的这套软件架构,我们自己在后面自己添加了一个 LCD 显示的部分。

在这里插入图片描述


2、mem_malloc_init

在这里插入图片描述
在这里插入图片描述

(1) mem_malloc_init 函数用来初始化 uboot 的堆管理器。

(2) uboot 中自己维护了一段堆内存,肯定自己就有一套代码来管理这个堆内存。有了这些东西 uboot 中你也可以 malloc、free 这套机制来申请内存和释放内存。我们在 DDR 内存中给堆预留了 896KB 的内存。


3、代码实践,去掉 Flash 看会不会出错。

结论:加上CONFIG_NOFLASH 宏之后编译出错,说明代码移植的不好,那个文件的包含没有被这个宏控制。于是乎移植的人就直接放这没管。


五、start_armboot 解析10

1、开发板独有初始化:mmc 初始化

在这里插入图片描述

(1) 从536 到 768 行,是开发板独有的初始化。意思是三星用一套 uboot 同时满足了好多个系列型号的开发板,然后在这里把不同开发板自己独有的一些初始化写到了这里。用 #if 条件编译配合 CONFIG_xxx 宏来选定特定的开发板。

(2) X210 相关的配置在 599 行到 632 行。
在这里插入图片描述


(3) mmc_initialize 看名字就应该是 MMC 相关的一些基础的初始化,其实就是用来初始化 SoC 内部的 SD/MMC 控制器的。函数在 uboot/drivers/mmc/mmc.c 里

(4) uboot 中对硬件的操作(譬如网卡、SD 卡···)都是借用的 linux 内核中的驱动来实现的,uboot 根目录底下有个 drivers 文件夹,这里面放的全都是从 linux 内核中移植过来的各种驱动源文件。

(5) mmc_initialize 是具体硬件架构无关的一个 MMC 初始化函数,所有的使用了这套架构的代码,都调用这个函数来完成 MMC 的初始化。mmc_initialize 中再调用 board_mmc_initcpu_mmc_init 来完成具体的硬件的 MMC 控制器初始化工作。

在这里插入图片描述


(6) cpu_mmc_init 在 uboot/cpu/s5pc11x/cpu.c 中,这里面又间接的调用了 drivers/mmc/s3c_mmcxxx.c 中的驱动代码来初始化硬件 MMC 控制器。这里面分层很多,分层的思想一定要有,否则完全就糊涂了。

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述


六、start_armboot 解析11

1、env_relocate

在这里插入图片描述

(1) env_relocate 是环境变量的重定位,完成从 SD 卡中将环境变量读取到 DDR 中的任务。

(2) 环境变量到底从哪里来?SD 卡中有一些(8个)独立的扇区作为环境变量存储区域的。但是我们烧录/部署系统时,我们只是烧录了 uboot 分区、kernel 分区和 rootfs 分区,根本不曾烧录 env 分区

所以当我们烧录完系统,第一次启动时 ENV 分区是空的,本次启动 uboot 尝试去 SD 卡的 ENV 分区读取环境变量时失败(读取回来后进行 CRC 校验时失败),我们 uboot 选择从 uboot 内部代码中设置的一套默认的环境变量出发来使用(这就是默认环境变量)。

这套默认的环境变量在本次运行时,会被读取到 DDR 中的环境变量中,然后被写入(也可能是你saveenv时写入,也可能是uboot设计了第一次读取默认环境变量后就写入)SD 卡的 ENV 分区。然后下次再次开机时,uboot 就会从 SD 卡的 ENV 分区读取环境变量到 DDR 中,这次读取就不会失败了。


(3) 真正的从 SD 卡到 DDR 中重定位 ENV 的代码是在 env_relocate_spec 内部的 movi_read_env 完成的。

在这里插入图片描述

在这里插入图片描述


七、start_armboot 解析12

1、IP地址、MAC地址的确定

在这里插入图片描述

(1) 开发板的 IP 地址是在 gd->bd 中维护的,来源于环境变量 ipaddr。 getenv 函数用来获取字符串格式的 IP 地址,然后用 string_to_ip 将字符串格式的 IP 地址转成字符串格式的点分十进制格式。

(2) IP 地址由 4 个 0-255 之间的数字组成,因此一个 IP 地址在程序中最简单的存储方法就是一个 unsigend int 。但是人类容易看懂的并不是这种类型,而是点分十进制类型(192.168.1.2)。这两种类型可以互相转换


2、devices_init

在这里插入图片描述

在这里插入图片描述

(1) devices_init 看名字就是设备的初始化。这里的设备指的就是开发板上的硬件设备。放在这里初始化的设备都是驱动设备,这个函数本来就是从驱动框架中衍生出来的。uboot 中很多设备的驱动是直接移植 linux 内核的(譬如网卡、SD卡),linux 内核中的驱动都有相应的设备初始化函数。linux 内核在启动过程中就有一个 devices_init (名字不一定完全对,但是差不多),作用就是集中执行各种硬件驱动的 init 函数。

(2) uboot 的这个函数其实就是从 linux 内核中移植过来的,它的作用也是去执行所有的从 linux 内核中继承来的那些硬件驱动的初始化函数


3、jumptable_init

在这里插入图片描述

(1) jumptable 跳转表,本身是一个函数指针数组,里面记录了很多函数的函数名。看这阵势是要实现一个函数指针到具体函数的映射关系,将来通过跳转表中的函数指针就可以执行具体的函数。这个其实就是在用 C 语言实现面向对象编程。在 linux 内核中有很多这种技巧。

(2) 通过分析发现跳转表只是被赋值从未被引用,因此跳转表在 uboot 中根本就没使用。

在这里插入图片描述


八、start_armboot 解析13

1、console_init_r

在这里插入图片描述

(1) console_init_f 是控制台的第一阶段初始化,console_init_r 是第二阶段初始化。实际上第一阶段初始化并没有实质性工作,第二阶段初始化才进行了实质性工作

(2) console_init_r 就是 console 的纯软件架构方面的初始化(说白了就是去给 console 相关的数据结构中填充相应的值),所以属于纯软件配置类型的初始化。

(4) uboot 的 console 实际上并没有干有意义的转化,它就是直接调用的串口通信的函数。所以用不用 console 实际并没有什么分别。(在 linux 内 console 就可以提供缓冲机制等不用 console 不能实现的东西)。

在这里插入图片描述


2、enable_interrupts

在这里插入图片描述

(1) 看名字应该是中断初始化代码。这里指的是 CPSR 中总中断标志位的使能。

(2) 因为我们 uboot 中没有使用中断,因此没有定义 CONFIG_USE_IRQ 宏,因此我们这里这个函数是个空壳子。

在这里插入图片描述


(3) uboot 中经常出现一种情况就是,根据一个宏是否定义了来条件编译决定是否调用一个函数内部的代码。uboot 中有 2 种解决方案来处理这种情况:方案一:在调用函数处使用条件编译,然后函数体实际完全提供代码。方案二:在调用函数处直接调用,然后在函数体处提供 2 个函数体,一个是有实体的一个是空壳子,用宏定义条件编译来决定实际编译时编译哪个函数进去。


3、loadaddr、bootfile 两个环境变量

在这里插入图片描述

(1) 这两个环境变量都是内核启动有关的,在启动 linux 内核时会参考这两个环境变量的值。


4、board_late_init

在这里插入图片描述

(1) 看名字这个函数就是开发板级别的一些初始化里比较晚的了,就是晚期初始化。所以晚期就是前面该初始化的都初始化过了,剩下的一些必须放在后面初始化的就在这里了。侧面说明了开发板级别的硬件软件初始化告一段落了。

(2) 对于 X210 来说,这个函数是空的。

在这里插入图片描述


九、start_armboot解析14

1、eth_initialize

在这里插入图片描述

(1) 看名字应该是网卡相关的初始化。这里不是 SoC 与网卡芯片连接时,SoC 这边的初始化,而是网卡芯片本身的一些初始化。

(2) 对于 X210(DM9000)来说,这个函数是空的。X210 的网卡初始化在 board_init 函数中,网卡芯片的初始化在驱动中

在这里插入图片描述

在这里插入图片描述


2、x210_preboot_init(LCD和logo显示)

在这里插入图片描述

(1) x210 开发板在启动起来之前的一些初始化,以及 LCD 屏幕上的 logo 显示。

在这里插入图片描述


3、check menukey to update from sd

在这里插入图片描述

(1) uboot 启动的最后阶段设计了一个自动更新的功能。就是:我们可以将要升级的镜像放到 SD 卡的固定目录中,然后开机时在 uboot 启动的最后阶段检查升级标志(是一个按键。按键中标志为 “LEFT” 的那个按键,这个按键如果按下则表示 update mode,如果启动时未按下则表示 boot mode)。如果进入 update mode,则 uboot 会自动从 SD 卡中读取镜像文件然后烧录到 iNand 中;如果进入 boot mode ,则 uboot 不执行 update,直接启动正常运行。

(2) 这种机制能够帮助我们快速烧录系统,常用于量产时用 SD 卡进行系统烧录部署。


4、死循环

在这里插入图片描述

(1) 解析器

在这里插入图片描述


(2) 开机倒数自动执行

在这里插入图片描述


(3) 命令补全

在这里插入图片描述


十、uboot 启动第 2 阶段总结

1、启动流程回顾、重点函数标出

(1) 第二阶段主要是对开发板级别的硬件、软件数据结构进行初始化。

(2) 代码流程顺序

    init_sequence
		cpu_init	空的
		board_init	网卡、机器码、内存传参地址
			dm9000_pre_init			网卡
			gd->bd->bi_arch_number	机器码
			gd->bd->bi_boot_params	内存传参地址
		interrupt_init	定时器
		env_init
		init_baudrate	gd数据结构中波特率
		serial_init		空的
		console_init_f	空的
		display_banner	打印启动信息
		print_cpuinfo	打印CPU时钟设置信息
		checkboard		检验开发板名字
		dram_init		gd数据结构中DDR信息
		display_dram_config	打印DDR配置信息表
	mem_malloc_init		初始化uboot自己维护的堆管理器的内存
	mmc_initialize		inand/SD卡的SoC控制器和卡的初始化
	env_relocate		环境变量重定位
	gd->bd->bi_ip_addr	gd数据结构赋值
	gd->bd->bi_enetaddr	gd数据结构赋值
	devices_init		空的
	jumptable_init		不用关注的
	console_init_r		真正的控制台初始化
	enable_interrupts	空的
	loadaddr、bootfile 	环境变量读出初始化全局变量
	board_late_init		空的
	eth_initialize		空的
	x210_preboot_init	LCD初始化和显示logo
	check_menu_update_from_sd	检查自动更新
	main_loop			主循环

2、启动过程特征总结

(1) 第一阶段为汇编阶段、第二阶段为 C 阶段;

(2) 第一阶段在 SRAM 中、第二阶段在 DRAM 中;

(3) 第一阶段注重 SoC 内部、第二阶段注重 SoC 外部 Board 内部;


3、移植时的注意点

(1) x210_sd.h 头文件中的宏定义;

(2) 特定硬件的初始化函数位置(譬如网卡)。


源自朱有鹏老师.

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

ARM uboot 源码分析5 -启动第二阶段 的相关文章

  • 如何获取 linux 实用程序 tail 的源代码?

    这个命令确实非常有用 但是我可以在哪里获取源代码以查看内部发生的情况 thanks tail 实用程序是 Linux 上 coreutils 的一部分 源压缩包 ftp ftp gnu org gnu coreutils coreutils
  • 如何用X11复制到剪贴板?

    使用 OS X 上的框架 我可以使用以下命令将 PNG 复制到粘贴板 在 C 中 显然我可以将 NSPasteboard 与 Cocoa 一起使用 include
  • 如何让 clangd 转向 c++20

    当没有其他信息时 如何让 clangd 回退到 c 20 例如 在第一次构建之前 cmake 可以生成一个 这是在带有最新 LLVM 的 Arch Linux 上 这是通过 Emacs LSP 运行的 但这应该没有什么区别 你可以加 Com
  • 停止服务时单元陷入故障状态(状态=143)[关闭]

    Closed 这个问题不符合堆栈溢出指南 help closed questions 目前不接受答案 这是我的问题 我有 CentOS 和 java 进程在上面运行 Java进程是通过启动 停止脚本来操作的 它也创建了 java 实例的 p
  • 在 Ubuntu 16.04 上找不到 printf.c

    我最近切换到Ubuntu 16 04 我在用vscode作为 Ubuntu 上的 IDE 我配置了其他语言 但我无法做到这一点C C 我创建c cpp properties json launch json tasks json 当我开始编
  • Bash 方法的返回值总是模 256

    我有一个 bash 脚本方法 它返回输入值 然而 返回值始终是模 256 的值 我用 google 搜索了一段时间 发现this http www tldp org LDP abs html exitcodes html文章说它总是以 25
  • 从 TypeScript 运行任何 Linux 终端命令?

    有没有办法直接从 TypeScript 类中执行 Linux 终端命令 这个想法是做类似的事情 let myTerminal new LinuxTerminal let terminalResult myTerminal run sudo
  • 劫持系统调用

    我正在编写一个内核模块 我需要劫持 包装一些系统调用 我正在暴力破解 sys call table 地址 并使用 cr0 来禁用 启用页面保护 到目前为止一切顺利 一旦完成 我将公开整个代码 因此如果有人愿意 我可以更新这个问题 无论如何
  • 如何查找连接到 AF_INET 套接字的客户端的 UID?

    有什么方法或类似的东西ucred for AF UNIX如果是AF INET插座 TCP在我的例子中 找出连接到我的套接字的客户端的UID 还有 proc net tcp但它显示了UID of the creator插座的而不是连接的cli
  • Linux shell 脚本中的 while 循环超时

    这工作正常 无限循环 while TRUE do printf done 我在尝试着timeout this while loop与timeout命令 所有这些都不起作用 timeout 5 while TRUE do printf don
  • 使用 NEON 内在函数除以浮点数

    我当时正在处理四个像素的图像 这是在armv7对于 Android 应用程序 我想分一个float32x4 t向量由另一个向量组成 但其中的数字与大约不同0 7 to 3 85 在我看来 除法的唯一方法是使用右移 但这是针对一个数字2 n
  • 如何使用 sed 仅删除双空行?

    我找到了这个问题和答案 https stackoverflow com questions 4651591 howto use sed to remove only triple empty lines关于如何删除三重空行 但是 我只需要对
  • 信号处理程序有单独的堆栈吗?

    信号处理程序是否有单独的堆栈 就像每个线程都有单独的堆栈一样 这是在 Linux C 环境中 来自 Linux 手册页signal 7 http kernel org doc man pages online pages man7 sign
  • 如何制作和应用SVN补丁?

    我想制作一个SVN类型的补丁文件httpd conf这样我就可以轻松地将其应用到其他主机上 If I do cd root diff Naur etc httpd conf httpd conf original etc httpd con
  • 并行运行 make 时出错

    考虑以下制作 all a b a echo a exit 1 b echo b start sleep 1 echo b end 当运行它时make j2我收到以下输出 echo a echo b start a exit 1 b star
  • 如何修复“iptables:没有该名称的链/目标/匹配”?

    我在我的 Linux 嵌入式系统上构建并安装了 iptables 如果我列出所有规则 则一切正常 iptables list Chain INPUT policy ACCEPT target prot opt source destinat
  • 拆分字符串以仅获取前 5 个字符

    我想去那个地点 var log src ap kernelmodule 10 001 100 但看起来我的代码必须处理 ap kernelmodule 10 002 100 ap kernelmodule 10 003 101 等 我想使用
  • 何时使用 pthread 条件变量?

    线程问题 看来 只有在其他线程调用 pthread cond notify 之前调用 pthread cond wait 时 条件变量才起作用 如果在等待之前发生通知 那么等待将被卡住 我的问题是 什么时候应该使用条件变量 调度程序可以抢占
  • 修改linux下的路径

    虽然我认为我已经接近 Linux 专业人士 但显然我仍然是一个初学者 当我登录服务器时 我需要使用最新版本的R 统计软件 R 安装在 2 个地方 当我运行以下命令时 which R I get usr bin R 进而 R version
  • 创建 jar 文件 - 保留文件权限

    我想知道如何创建一个保留其内容的文件权限的 jar 文件 我将源代码和可执行文件打包在一个 jar 文件中 该文件将在使用前提取 人们应该能够通过运行批处理 shell 脚本文件立即运行示例和演示 然后他们应该能够修改源代码并重新编译所有内

随机推荐

  • OpenCV——分水岭算法

    目录 一 分水岭算法 1 概述 2 图像分割概念 3 分水岭算法原理 二 主要函数 三 C 代码 四 结果展示 1 原始图像 2 分割结果 五 参考链接 一 分水岭算法 1 概述 分水岭算法是一种图像分割常用的算法 可以有效地将图像中的目标
  • Javascript高级程序设计——15-1.匿名函数和闭包

    1 匿名函数 表示没有定义函数名的函数 案例1 1 简单的匿名函数 function 单独的匿名函数无法执行 alert Lee 案例1 2 将匿名函数赋值给一个变量 var box function return Lee alert bo
  • 复数矩阵计算行列式

    项目上需要对复矩阵的行列式计算 根据计算一般矩阵行列式的代码改成了复矩阵行列式计算 include
  • 性能测试中TPS上不去的几种原因

    中TPS一直上不去 是什么原因 这篇文章 就具体说说在实际压力测试中 为什么有时候TPS上不去的原因 先来解释下什么叫TPS TPS Transaction Per Second 每秒事务数 指服务器在单位时间内 秒 可以处理的事务数量 一
  • Python库的使用说明

    目录 1 第三方库索引网站 2 第三方安装 2 1 pip工具介绍 2 2 pip工具安装 2 2 1 list 命令查看已安装的库列表 2 2 2 uninstall 命令 2 2 3 show 命令 2 2 4 download 命令
  • C++标准模板库 迭代器 iterator 详解(二)

    迭代器提供对一个容器中的对象的访问方法 并且定义了容器中对象的范围 迭代器就如同一个指针 事实上 C 的指针也是一种迭代器 但是 迭代器不仅仅是指针 因此你不能认为他们一定具有地址值 例如 一个数组索引 也可以认为是一种迭代器 迭代器有各种
  • [NOI2009]植物大战僵尸【拓扑+最大权闭合子图】

    题目链接 BZOJ 1565 看到这道题之后很容易想到的就是最大权闭合子图了 但是却有个问题就是要去除掉那些环 因为构成了环之后 相当于是无敌的状态 它们就永远不会得到贡献 并且环之后的点也是得不到贡献的 所以 这里利用拓扑 知道哪些点是可
  • 「Qt」事件概念

    0 引言 在本文所属专栏的前面的文章里 我们介绍了Qt的 信号 Signal 与 槽 Slot 机制 信号 Signal 与 槽 Slot 机制是 Qt 框架用于多个对象之间通信的 是 Qt 的核心特性 也是 Qt 与其他框架最大的不同之处
  • anaconda中spyder改变背景颜色(黑色)

    spyder挺好用的 但是未定义的背景颜色实在不好看 纯属个人审美 下面开始更换背景图 打开spyder 依此点击 Tools 再点击preference 喜爱 选择Syntax coloring Scheme调成Monokai 这是我喜欢
  • python+selenium+unittest自动化测试框架

    前言 关于自动化测试的介绍 网上已有很多资料 这里不再赘述 UI自动化测试是自动化测试的一种 也是测试金字塔最上面的一层 selenium是应用于web的自动化测试工具 支持多平台 多浏览器 多语言来实现自动化 优点如下 开源 免费且对we
  • pyecharts在数据可视化中的应用 (二)(pyecharts绘制树图、矩形树图、地理热力图、词云图、相关性矩阵等图)

    1 使用以下JSON数据绘制树图 矩形树图 from pyecharts import options as opts from pyecharts charts import Tree data name flare children n
  • Android 系统性能优化(57)---MTK 平台开关机、重启时间优化

    MTK 平台开关机 重启时间优化 开关机 重启时间优化 开机性能优化 是用功能和其它因素多方面平衡的结果 片面追求单方面的性能没有太大意义 有些产品设计开机动画非常酷炫 动画图片过多 高帧率会影响开机速度 这时就需要看是开机速度优先还是体验
  • 人工智能(pytorch)搭建模型8-利用pytorch搭建一个BiLSTM+CRF模型,实现简单的命名实体识别

    大家好 我是微学AI 今天给大家介绍一下人工智能 pytorch 搭建模型8 利用pytorch搭建一个BiLSTM CRF模型 实现简单的命名实体识别 BiLSTM CRF 模型是一种常用的序列标注算法 可用于词性标注 分词 命名实体识别
  • kubernetes资源控制器【一】- ReplicaSet控制器

    一 Pod控制器 Master的各组件中 API Server仅负责将资源存储于etcd中 并将其变动通知给各相关的客户端程序 如kubelet kube scheduler kube proxy和kube controller manag
  • id和instancetype的应用场景区别

    在 Objective C 中 id 是一个通用的指针类型 可以用来表示任何类型的对象 而instancetype是一个表示当前类类型的指针类型 通常用于方法的返回值类型 下面是它们的一些使用场景 使用id的情况 当你需要一个指向任何对象的
  • ubuntu 触摸板失灵解决

    ubuntu 触摸板失灵解决 Ubuntu 20 04 开机发现触摸板只能单击 经常漂移影响打字输入 操作 sudo modprobe r psmouse sudo modprobe psmouse 目的在于重新加载内核触摸板模块 重新加载
  • jquery ui 实现table的sortable功能以及过滤记录功能

    本人在工作中曾使用js实现过用鼠标拖动表格的行实现重新排序的功能 当时写了不少的js代码 最近发现jquery ui也能实现这个功能 而且很方便 真后悔当时不知道有这么个好东东 好 现在介绍下如何使用jquery ui来实现 引入的js文件
  • 邻结矩阵的创建

    图的邻结矩阵是储存图数据的一个手段 储存方式是用两个数组来表示圆 一个一维数组储存图中的顶点信息 一个二维数组 称为邻结矩阵 储存图中边或弧的信息 代码展示 include
  • Kotlin筑基

    Kotlin筑基 本文链接 核心思路 每个知识点都要和源码结合起来讲 文章目录 Kotlin筑基 编译时常量 基本类型 range 访问权修饰符 Unit Nothing 反引号 函数内联 函数引用 具名函数 判空和安全调用 断言操作 空合
  • ARM uboot 源码分析5 -启动第二阶段

    一 start armboot 解析6 1 console init f 1 console init f 是 console 控制台 的第一阶段初始化 f 表示是第一阶段初始化 r 表示第二阶段初始化 有时候初始化函数不能一次一起完成 中