52、U-boot2023的移植教程

2024-06-23 14:12
文章标签 52 boot2023 教程 移植

本文主要是介绍52、U-boot2023的移植教程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

uboot:https://ftp.denx.de/pub/u-boot/
nxp-uboot:https://github.com/nxp-imx/uboot-imx

1、顶层Makefile

 文件加入编译的两种方式:以xxx/xxx.c文件为例
        1、使用menuconfig:
              先编辑.c所在目录下的Kconfig,加入配置项xxx
              再编辑.c所在目录下的Makefile,添加obj-$(CONFIG_xxx) = xxx.o
        2、不使用menuconfig:
              直接编辑.c所在目录下的Makefile,添加obj-y = xxx.o
        说明:$(libs-y)依赖每个文件夹下的xxx-in.o,而每个文件夹下的xxx-in.o又依赖当前文件夹的所有.o文件。

2、Kbuild框架

<1>在scripts文件夹下,有一个Kbuild.include文件中定义了如下几个关键的变量:181: build := -f $(srctree)/scripts/Makefile.build obj187: modbuiltin := -f $(srctree)/scripts/Makefile.modbuiltin obj193: dtbinst := -f $(srctree)/scripts/Makefile.dtbinst obj199: clean := -f $(srctree)/scripts/Makefile.clean obj205  hdr-inst := -f $(srctree)/scripts/Makefile.headersinst obj217: echo-cmd = $(if $($(quiet)cmd_$(1)),\echo '  $(call escsq,$($(quiet)cmd_$(1)))$(echo-why)';).....<2>在顶层Makefile文件的391-393行有如下:# We need some generic definitions (do not try to remake the file).scripts/Kbuild.include: ;include scripts/Kbuild.include                   # 包含<1>中定义的变量<3>在顶层Makefile文件的1232-1234行有如下:%.imx: $(IMX_DEPS) %.bin$(Q)$(MAKE) $(build)=arch/arm/mach-imx $@    # <1>中定义的build被使用$(BOARD_SIZE_CHECK)<4>在顶层Makefile文件的1255-1259行有如下:(make dtbs命令)PHONY += dtbsdtbs: dts/dt.dtb@:dts/dt.dtb: u-boot$(Q)$(MAKE) $(build)=dts dtbs  展开得:@ make -f $(srctree)/scripts/Makefile.build obj=dts dtbs其中:obj=dts是一个传入的变量, dtbs是要构建的目标

2、u-boot.map文件

反汇编:arm-linux-gnueabihf-objdump -D -m arm u-boot > u-boot.dis

./scripts/dtc/dtc -I dtb -O dts -o ./itop.dts ./arch/arm/dts/imx6ull-14x14-itop.dtb

./scripts/dtc/dtc -I dtb -O dts -o ./evk.dts ./arch/arm/dts/imx6ull-14x14-evk.dtb

注意:如何确定哪个.c文件的哪个函数被编译?
        通过函数名在u-boot.map、u-boot.dis这两个文件中搜索。先在u-boot.dis找到函数的链接地址,以此链接地址在u-boot.map中搜索,即可确定是哪个.c文件下的函数被编译。

save_boot_params_ret:/* disable interrupts (FIQ and IRQ), also set the cpu to SVC32 mode, except if in HYP mode already *//* 禁用中断(FIQ 和 IRQ),同时将处理器设置为 SVC32 模式,除非已经处于 HYP 模式。 */mrs	r0, cpsr            @ 将当前程序状态寄存器(CPSR)的值加载到寄存器r0中and	r1, r0, #0x1f		@ 使用掩码操作提取r0的低5位,即当前的处理器模式teq	r1, #0x1a		    @ 测试是否当前处理器模式为HYP模式(Hypervisor mode)bicne	r0, r0, #0x1f	@ 如果不在HYP模式,清除 CPSR 中的所有模式位orrne	r0, r0, #0x13	@ 如果不在HYP模式,设置 CPSR 的模式为SVC模式orr	r0, r0, #0xc0		@ 禁用FIQ和IRQ中断,将 CPSR 的FIQ和IRQ位设置为 1msr	cpsr,r0             @ 将修改后的 CPSR 的值写回 CPSR 寄存器,完成设置/* Set V=0 in CP15 SCTLR register - for VBAR to point to vector *//* 将 CP15 的 SCTLR 寄存器中将 V 位设置为0,以便将 VBAR 指向向量表 */mrc	p15, 0, r0, c1, c0, 0	@ 读取 CP15 SCTLR 寄存器里面的值bic	r0, #CR_V		        @ 将 CP15 SCTLR 寄存器中的 V 位清零mcr	p15, 0, r0, c1, c0, 0	@ 写回修改后的值到 CP15 SCTLR 寄存器/* Set vector address in CP15 VBAR register *//* 在 CP15 的 VBAR 寄存器中设置向量表的地址   */ldr	r0, =_start             @ 将 _start 符号的地址加载到寄存器 r0 中mcr	p15, 0, r0, c12, c0, 0	@ 将 r0 中的地址写入 CP15 VBAR 寄存器,设置异常向量表基地址bl	cpu_init_cp15bl	cpu_init_critbl	_main注:对于CP15协处理器的操作,CP15有c0-c15共16个寄存器组,每个组里面含有多个寄存器
mcr	p15, 0, r0, c12, c0, 0  @读操作,读取c12寄存器组里面的c0,0对应的寄存器到r0中
注意:c0,0并不是偏移的意思,而是由具体的表决定的,不要以为c0,1就是组里的第1个寄存器,这是不一定的。具体的cx,x对应组里面的第几个寄存器,参考手册上有写。
/************************************************************************************** cpu_init_cp15* 设置 CP15 寄存器(缓存、MMU、TLBs)。如果定义了 CONFIG_SYS_ICACHE_OFF,则打开 I-cache。*************************************************************************************/
cpu_init_cp15:/* Invalidate L1 I/D */mov	r0, #0			        @ set up for MCR (r0 = 0)mcr	p15, 0, r0, c8, c7, 0	@ invalidate TLBs (置零)mcr	p15, 0, r0, c7, c5, 0	@ invalidate icache (置零)mcr	p15, 0, r0, c7, c5, 6	@ invalidate BP array (置零)dsbisb/* disable MMU stuff and caches */mrc	p15, 0, r0, c1, c0, 0  @ 读 SCTLR => r0 bic	r0, r0, #0x00002000	   @ clear bits 13 (--V-)bic	r0, r0, #0x00000007	   @ clear bits 2:0 (-CAM)orr	r0, r0, #0x00000002	   @ set bit 1 (--A-) Alignorr	r0, r0, #0x00000800	   @ set bit 11 (Z---) BTBorr	r0, r0, #0x00001000	   @ set bit 12 (I) I-cachemcr	p15, 0, r0, c1, c0, 0  @ 写 r0 => SCTLRmov	r5, lr			       @ Store my Caller (r5 = lr)(跳转)mrc	p15, 0, r1, c0, c0, 0  @ r1 has Read Main ID Register (r1 = MIDR)mov	r3, r1, lsr #20		   @ get variant field (r3 = r1>>20)and	r3, r3, #0xf		   @ r3 has CPU variant (r3 = r3&0xf)and	r4, r1, #0xf		   @ r4 has CPU revision (r4 = r1&0xf)mov	r2, r3, lsl #4		   @ shift variant field for combined value(r2 = r3<<4)orr	r2, r4, r2		       @ r2 has combined CPU variant + revision(r2 = r4|r2)/* Early stack for ERRATA that needs into call C code */
#if defined(CONFIG_SPL_BUILD) && defined(CONFIG_SPL_STACK)ldr	r0, =(CONFIG_SPL_STACK)
#elseldr	r0, =(SYS_INIT_SP_ADDR)
#endifbic	r0, r0, #7	 @ 8-byte alignment for ABI compliance(将r0最低的三个比特位清零)mov	sp, r0       @ SP = r0mov	pc, r5	     @ back to my caller ( pc=r5 )(返回)
cpu_init_crit:b lowlevel_init   #(注意:这是一个无返回跳转)# 下面具体分析lowlevel_init -------------------------------------------------
lowlevel_init:/*Setup a temporary stack. Global data is not available yet. */
#if defined(CONFIG_SPL_BUILD) && defined(CONFIG_SPL_STACK)ldr	sp, =CONFIG_SPL_STACK  @ 设置堆栈指针
#elseldr	sp, =SYS_INIT_SP_ADDR  @ 设置堆栈指针
#endifbic	sp, sp, #7    @ 8字节对齐 for ABI compliance(将sp最低的三个比特位清零) 
#ifdef CONFIG_SPL_DMmov	r9, #0        @ r9 = 0
#elsepush  {ip, lr}    @ 压ip入栈、压lr入栈bl	s_init        @ 调用s_init(C函数)初始化处理器的时钟pop	{ip, pc}      @ 出栈#include <asm/io.h>
#include <asm/arch/imx-regs.h>
_main:
#if defined(CONFIG_TPL_BUILD) && defined(CONFIG_TPL_NEEDS_SEPARATE_STACK)ldr	r0, =(CONFIG_TPL_STACK)
#elif defined(CONFIG_SPL_BUILD) && defined(CONFIG_SPL_STACK)ldr	r0, =(CONFIG_SPL_STACK)
#elseldr	r0, =(SYS_INIT_SP_ADDR)
#endifbic	r0, r0, #7	/* 8-byte alignment for ABI compliance */mov	sp, r0                       @ sp = r0 (设置栈指针)bl	board_init_f_alloc_reserve   @ 在栈中,分配早期的malloc区域和gd区域(r0为函数的参数)mov	sp, r0                       @ sp = r0 (r0为函数的返回值)/* set up gd here, outside any C code */mov	r9, r0                       @ r9 = r0bl	board_init_f_init_reserve    @ 对早期的malloc区域和gd区域进行初始化(r0为函数的参数)#if defined(CONFIG_DEBUG_UART) && CONFIG_IS_ENABLED(SERIAL)bl	debug_uart_init
#endif#if defined(CONFIG_SPL_BUILD) && defined(CONFIG_SPL_EARLY_BSS)CLEAR_BSS
#endifmov	r0, #0bl	board_init_fldr	r0, [r9, #GD_START_ADDR_SP]	/* sp = gd->start_addr_sp */bic	r0, r0, #7	/* 8-byte alignment for ABI compliance */mov	sp, r0ldr	r9, [r9, #GD_NEW_GD]		/* r9 <- gd->new_gd */adr	lr, here......bl relocate_code......
struct driver *drv = ll_entry_start(struct driver, driver);struct driver *drv = ({									\static char start[0] __aligned(CONFIG_LINKER_LIST_ALIGN)	\__attribute__((unused))					\__section("__u_boot_list_2_driver_1");			\struct driver * tmp = (struct driver *)&start;					\asm("":"+r"(tmp));						\tmp;								\
});//说明:返回__u_boot_list_2_driver_1段的第一个元素的起始地址

3.1、添加自己的单板<主>

1、添加头文件cp ./include/configs/mx6ullevk.h   ./include/configs/mx6ullitop.hvim  ./include/configs/mx6ullitop.h修改.h开头的宏定义2、添加板级文件夹cp -r ./board/freescale/mx6ullevk   ./board/freescale/mx6ullitopmv ./board/freescale/mx6ullitop/mx6ullevk.c ./board/freescale/mx6ullitop/mx6ullitop.cvim ./board/freescale/mx6ullitop/Kconfigvim ./board/freescale/mx6ullitop/Makefilevim ./board/freescale/mx6ullitop/MAINTAINERSvim ./board/freescale/mx6ullitop/imximage.cfg3、添加设备树文件1cp  ./arch/arm/dts/imx6ull-14x14-evk.dts  ./arch/arm/dts/imx6ull-14x14-itop.dtsvim ./arch/arm/dts/imx6ull-14x14-itop.dtsvim ./arch/arm/dts/Makefile在 dtb-$(CONFIG_MX6ULL) 添加一项imx6ull-14x14-itop.dtb \   (为了使能dtb的编译)4、添加设备树文件2 cp ./arch/arm/dts/imx6ull-14x14-evk-u-boot.dtsi ./arch/arm/dts/imx6ull-14x14-itop-u-boot.dtsi5、添加默认配置文件cp ./configs/mx6ull_14x14_evk_defconfig ./configs/mx6ull_14x14_itop_defconfigvim configs/mx6ull_14x14_itop_defconfig<1>找到 CONFIG_TARGET_MX6ULL_14X14_EVK=y​<1>改为 CONFIG_TARGET_MX6ULL_14X14_ITOP=y<2>找到 CONFIG_DEFAULT_DEVICE_TREE="imx6ull-14x14-evk"<2>改为 ​CONFIG_DEFAULT_DEVICE_TREE="imx6ull-14x14-itop"​​6、修改Kconfig文件vim ./arch/arm/mach-imx/mx6/Kconfig添加config TARGET_MX6ULL_14X14_ITOPbool "Support mx6ull_14x14_itop"depends on MX6ULLselect BOARD_LATE_INITselect DMselect DM_THERMALselect MX6ULLimply CMD_DM添加source "board/freescale/mx6ullitop/Kconfig"
7、修改 ./arch/arm/mach-imx/cpu.c,在reset_cpu函数中添加如下代码:方法一:注释掉此函数,此时会链接drivers/watchdog/imx_watchdog.c中的reset_cpu函数
/*
__weak void reset_cpu(void)
{return;
}
*/
方法二:修改函数如下
__weak void reset_cpu(void)
{#include <fsl_wdog.h>struct watchdog_regs *wdog = (struct watchdog_regs *)WDOG1_BASE_ADDR;//do not assert internal resetu16 wcr = 0x04|0x10; // WCR_WDE|WCR_SRS //Write 3 times to ensure it works, due to IMX6Q errata ERR004346 writew(wcr, &wdog->wcr);writew(wcr, &wdog->wcr);writew(wcr, &wdog->wcr);//Start while for(;;);return;
}

3.2、移植网络的驱动<主>

< 1 >设备树:imx6ull-14x14-itop.dts

一、设备树:目录
[arch/arm/dts/imx6ull-14x14-itop.dts]|-->#include "imx6ull.dtsi"|-->#include "imx6ul.dtsi"|-->#include <dt-bindings/clock/imx6ul-clock.h>|-->#include <dt-bindings/gpio/gpio.h>|-->#include <dt-bindings/input/input.h>|-->#include <dt-bindings/interrupt-controller/arm-gic.h>|-->#include "imx6ul-pinfunc.h"|-->#include "imx6ull-pinfunc.h"|-->#include "imx6ull-pinfunc-snvs.h"|-->#include "imx6ul-14x14-evk.dtsi"
[arch/arm/dts/imx6ull-14x14-evk-u-boot.dtsi]
二、设备树:网络相关
[arch/arm/dts/imx6ull-14x14-itop.dts]|-->#include "imx6ull.dtsi"|-->#include "imx6ul.dtsi"//Fast Ethernet Controller 1 (CPU内部资源)
&fec1 { pinctrl-names = "default";pinctrl-0 = <&pinctrl_enet1 &pinctrl_enet1_reset>; //zjh addphy-mode = "rmii";phy-handle = <&ethphy0>;phy-supply = <&reg_peri_3v3>;phy-reset-gpios = <&gpio5 7 GPIO_ACTIVE_LOW>; //zjh addphy-reset-duration = <200>;                   //zjh addstatus = "okay";
};
//Fast Ethernet Controller 2 (CPU内部资源)
&fec2 { pinctrl-names = "default";pinctrl-0 = <&pinctrl_enet2 &pinctrl_enet2_reset>; //zjh addphy-mode = "rmii";phy-handle = <&ethphy1>;phy-supply = <&reg_peri_3v3>;phy-reset-gpios = <&gpio5 8 GPIO_ACTIVE_LOW>; //zjh addphy-reset-duration = <200>;                   //zjh addstatus = "okay";mdio { //MDIO总线:用于连接 FEC 和 PHY 设备#address-cells = <1>;#size-cells = <0>;ethphy0: ethernet-phy@2 { //(网口1)compatible = "ethernet-phy-id0022.1560";reg = <2>;                //phy芯片地址(硬件电路决定)micrel,led-mode = <1>;clocks = <&clks IMX6UL_CLK_ENET_REF>;clock-names = "rmii-ref";};ethphy1: ethernet-phy@1 { //(网口2)compatible = "ethernet-phy-id0022.1560";reg = <1>;                //phy芯片地址(硬件电路决定)micrel,led-mode = <1>;clocks = <&clks IMX6UL_CLK_ENET2_REF>;clock-names = "rmii-ref";};};
};			

< 2 >迅为开发板硬件原理图

5、DM框架

DM 是 U-Boot 中的驱动框架,全称 Driver Mode。
Linux 中 platform bus 模型的驱动有着三要素:device 、bus 、driver 
Uboot 中 Driver Mode 模型的驱动也有三要素:udevice、uclass、driver。

1、udevice 描述具体的某一个硬件设备。struct udevice {const struct driver *driver;const char *name;......};通过三种路径生成:a、dts设备节点。(大多数使用)b、U_BOOT_DEVICE(__name) 宏申明 (少部分使用)c、主动调用 device_bind_xxx 系列 API (极少部分使用)2、driver 是与这个设备匹配的驱动。通过 U_BOOT_DRIVER(__name) 宏声明。如果 driver 实现了 bind 接口,该 bind 将在 device_bind_common 中 device 和 driver 匹配上后被调用, 而且在 device_bind_common 中会完成 udevice 和 driver 的绑定。driver 一般都有对应的 probe 接口,通过 device_probe(struct udevice *dev) 调用,要注意的是driver 的 bind 接口调用的比 probe 接口早, 大部分在 dm_init_and_scan 中就被调用了driver 一般会提供 ops 操作接口,供上一层调用。需要说明的是,driver 一般都不需要把自己注册到 uclass 中,而是在 device_bind_common 阶段实现 driver 、uclass、device 三者的对接,然后 uclass 层通过 udevice->driver->ops 获取对应 driver 的操作接口。3、uclass 是同一类设备的抽象,提供管理同一类设备的抽象接口主要包括两个类型的结构体:struct uclass_driver 和 struct uclass其中struct uclass_driver 为 struct uclass 的驱动struct uclass_driver {  //由UCLASS_DRIVER(__name)定义const char *name; enum uclass_id id;int (*post_bind)(struct udevice *dev);......};struct uclass {void *priv_;                  //类本身私有数据struct uclass_driver *uc_drv; //类本身的驱动程序struct list_head dev_head;    //该类中的设备列表struct list_head sibling_node;//类链表中的下一个类};

uclass和udevice都是动态生成的

1、在解析设备树中的设备或直接定义的平台设备的时候,会动态生成udevice。
2、然后找到udevice对应的driver,通过driver中的uclass id得到uclass_driver id。
3、从uclass链表中查找对应的uclass是否已经生成,没有生成的话则动态生成uclass。
注:uclass链表的起点是gd的一个成员:gd->uclass_root        (*全局变量gd的很重要*)

设备驱动的使用(应用层使用驱动)

1、首先需要通过 uclass_get_device_xxx 系列 API 拿到该设备的 udevice。
2、然后通过该设备的 uclass 提供的 API 操作该设备。
<+>uclass_get_device_xxx 拿到该设备的 udevice 后会调用该设备的 probe 接口。
以驱动[ pwm backlight ]为例:

/*** drivers/video/simple_panel.c*/
struct udevice *bldev;
uclass_get_device_by_phandle(UCLASS_PANEL_BACKLIGHT, dev, "backlight", &bldev);
backlight_enable(bldev);
backlight_set_brightness(bldev, percent);

x、启动分析(基于IMX6ULL)

u-boot2023:启动详细的代码调用流程
u-boot.lds:  [arch/arm/cpu/u-boot.lds]
-->_start:  [arch/arm/lib/vectors.S]-->reset:  [arch/arm/cpu/armv7/start.S]    -->save_boot_params:  [arch/arm/cpu/armv7/start.S] /*将引导参数保存到内存中*/-->save_boot_params_ret:   [arch/arm/cpu/armv7/start.S]|-->cpu_init_cp15:      [arch/arm/cpu/armv7/start.S]|-->cpu_init_crit:      [arch/arm/cpu/armv7/start.S]|-->lowlevel_init:  [arch/arm/cpu/armv7/lowlevel_init.S]|-->ENTRY(_main)              [arch/arm/lib/crt0.S]|-->board_init_f_alloc_reserve [common/init/board_init.c) /*为u-boot的gd结构体分配空间*/|-->board_init_f_init_reserve  [common/init/board_init.c) /*将gd结构体清零*/|-->board_init_f               [common/board_f.c]|-->initcall_run_list      [include/initcall.h]       /*初始化序列函数*/|-->init_sequence_f[]  [common/board_f.c]         /*初始化序列函数数组 */|-->setup_mon_len|-->fdtdec_setup|-->initf_malloc|-->log_init|-->...|-->arch_cpu_init|-->mach_cpu_init|-->initf_dm|-->board_early_init_f|-->...|-->env_init|-->init_baud_rate|-->serial_init|-->console_init_f|-->...|-->dram_init|-->...---------------------------------------------------------------------------分界线							|-->relocate_code       [arch/arm/lib/relocate.S]    /*主要完成镜像拷贝和重定位*/---------------------------------------------------------------------------分界线|-->relocate_vectors                        [arch/arm/lib/relocate.S]  /*重定位向量表*/|-->board_init_r                            [common/board_r.c]         /*重定向后板级初始化*/|-->initcall_run_list(init_sequence_r)  [include/initcall.h]       /*初始化序列函数*/|-->init_sequence_r[]               [common/board_r.c]         /*初始化序列函数数组*/|-->...|-->initr_dm					/*DM初始化*/ |-->board_init                  |-->...|-->initr_dm_devices            /*DM设备初始化*/|-->stdio_init_tables|-->serial_initialize           /*串口初始化*/|-->initr_announce					      |-->dm_announce                 |-->...	|-->initr_mmc                   /*MMC初始化*/|-->initr_env                   /*环境初始化*/|-->...	|-->interrupt_init|-->board_late_init|-->initr_net                   /*网络初始化*/|-->...|-->run_main_loop     [common/board_r.c] /* It does not return */								

这篇关于52、U-boot2023的移植教程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1087380

相关文章

Steam邮件推送内容有哪些?配置教程详解!

Steam邮件推送功能是否安全?如何个性化邮件推送内容? Steam作为全球最大的数字游戏分发平台之一,不仅提供了海量的游戏资源,还通过邮件推送为用户提供最新的游戏信息、促销活动和个性化推荐。AokSend将详细介绍Steam邮件推送的主要内容。 Steam邮件推送:促销优惠 每当平台举办大型促销活动,如夏季促销、冬季促销、黑色星期五等,用户都会收到邮件通知。这些邮件详细列出了打折游戏、

X-AnyLabeling使用教程

1.AI 模型自动分割标注使用教程 2.AI 模型自动目标检测标注使用教程

青龙面板2.9之Cdle傻妞机器人编译教程

看到有的朋友对傻妞机器人感兴趣,这里写一下傻妞机器人的编译教程。 第一步,这里以linux amd64为例,去官网下载安装go语言安装包: 第二步,输入下方指令 cd /usr/local && wget https://golang.google.cn/dl/go1.16.7.linux-amd64.tar.gz -O go1.16.7.linux-amd64.tar.gz

青龙面板部署通用教程,含服务器、路由器、X86等部署方法

1. 拉取镜像/更新镜像 docker pull whyour/qinglong:latest 2. 删除镜像 docker rmi whyour/qinglong:latest 3. 启动容器 普通服务器 docker run -dit \-v $PWD/ql/config:/ql/config \-v $PWD/ql/log:/ql/log \-v $PWD/ql/db:

宝塔面板部署青龙面板教程【简单易上手】

首先,你得有一台部署了宝塔面板的服务器(自己用本地电脑也可以)。 宝塔面板部署自行百度一下,很简单,这里就不走流程了,官网版本就可以,无需开心版。 首先,打开宝塔面板的软件商店,找到下图这个软件(Docker管理器)安装,青龙面板还是安装在docker里,这里依赖宝塔面板安装和管理docker。 安装完成后,进入SSH终端管理,输入代码安装青龙面板。ssh可以直接宝塔里操作,也可以安装ssh连接

PS系统教程25

介绍软件 BR(bridge) PS 配套软件,方便素材整理、管理素材 作用:起到桥梁作用 注意:PS和BR尽量保持版本一致 下载和安装可通过CSDN社区搜索,有免费安装指导。 安装之后,我们打开照片只需双击照片,就自动在Ps软件中打开。 前提:电脑上有PS软件 三种预览格式 全屏预览 评星级 直接按数字键就可以 方向键可以更换图片 esc退出 幻灯片放

flex布局学习笔记(flex布局教程)

前端笔试⾯试经常会问到:不定宽⾼如何⽔平垂直居中。最简单的实现⽅法就是flex布局,⽗元素加上如下代码即 可: display: flex; justify-content: center; align-items :center; 。下⾯详细介绍下flex布局吧。   2009年,W3C提出了 Flex布局,可以简便⼂完整⼂响应式地实现各种页⾯布局。⽬前已得到了所有浏览器的⽀持,这意味着,现

IPython小白教程:提升你的Python交互式编程技巧,通俗易懂!

IPython是一个增强的Python交互式shell,它提供了丰富的功能和便捷的交互方式,使得Python开发和数据分析工作更加高效。本文将详细介绍IPython的基本概念、使用方法、主要作用以及注意事项。 一、IPython简介 1. IPython的起源 IPython由Fernando Pérez于2001年创建,旨在提供一个更高效的Python交互式编程环境。 2. IPyt

信息检索(52):From doc2query to docTTTTTquery

From doc2query to docTTTTTquery 摘要 发布时间(2019) 摘要 Nogueira 等人 [7] 使用简单的序列到序列转换器 [9] 进行文档扩展。我们用 T5 [8] 替换转换器,并观察到效率大幅提升。 doc2query [7] 是一种文档扩展形式,其理念是训练一个模型,当给定一个输入文档时,该模型会生成该文档可能回答的问题。然后

图形编辑器基于Paper.js教程03:认识Paper.js中的所有类

先来认一下Paper的资源对象,小弟有哪些,有个整体的认识。认个脸。 在Paper.js的 官方文档中类大致有如下这些: 基类: ProjectViewItemPointToolSizeSegmentRectangleCurveCurveLocationMatrixColorStyleTweenToolEventGradientGradientStopEvent 二级或三级类 继承Ite