RISCVLinuxkernel启动代码分析之十一:设备树介绍(2)

原创 嵌入式Lee 2024-12-06 08:05

中断和中断映射

这是非常重要的部分

中断逻辑上构成一个树形结构

比如有中断产生的源头(设备产生中断)

interrupt specifier  描述中断号,中断电平

interrupt domain    中断产生设备

interrupt-controller  中断控制器

Interrupt Nexus  中断连接器,将子中断连接到父中断中去,一般用于PCI

中断树结构示意

中断发生设备的属性

interrupts

属性名: interrupts

值类型: 编码为任意数量的中断说明符

定义了设备生成的一个或多个中断。中断属性的值由任意数量的中断说明符组成。中断说明符的格式由绑定的interrupt domain root.定义。

interruptsinterrupts-extended 覆盖,二者只使用其一.

示例:

如下open PICcompatible interrupt domain

0x0A表示中断号,8表示中断电平的编码

interrupts = <0xA 8>;

interrupt-parent

属性名: interrupt-parent

值类型:

因为中断树中节点的层次结构可能与设备树不匹配,所以中断父属性可用于显式定义中断父属性。该值是中断父级的phandle。如果设备缺少此属性,则假定其中断父级是其设备树父级。

interrupts-extended

属性名: interrupts-extended

值类型:  

列出了设备生成的中断。当一个设备连接到多个中断控制器时,应该使用interrupts-extended来代替interrupts,因为其每个中断说明符都有对应的phandle

中断控制器的属性

#interrupt-cells

属性名: #interrupt-cells

值类型:

定义对interrupt domain的中断说明符 进行编码所需的单元.

interrupt-controller

属性名: interrupt-controller

值类型:

表示该节点是中断控制器节点

中断连接点Nexus属性

中断连接节点必须要#interrupt cell属性。

interrupt-map

属性名: interrupt-map

值类型: 编码为任意数量的中断映射条目interrupt mapping entries.

该属性用于将一个中断域interrupt domain与一组父中断域连接起来,并指定如何将子域中的中断说明符映射到它们各自的父域。

中断映射interrupt map 是一个表,其中每一行都是一个映射条目mapping entry,由五个部分组成: child unit

address子单元地址, child interrupt specifier子中断说明符, 中断interrupt-parent, parent unit address父单元地址,parent interrupt specifier父中断说明符。

child unit address:

映射的子节点的单元地址。表示该属性值需要的32位单元数由子节点所在的总线节点的#address-cells指定。

child interrupt specifier:

映射的子节点中断说明符。表示该属性值需要的32位单元数由本连接节点即interrupt-map属性所在节点的#interrupt-cells 决定。

interrupt-parent:

一个单 值用于执行子节点需要映射的父中断.

parent unit address:

父中断的节点地址,表示该属性值需要的32位单元数由 interrupt-parent指向的节点的 #address-cells属性指定。

parent interrupt specifier:

父节点中断说明符,表示该属性值需要的32位单元数由 interrupt-parent指向的节点的 #interrupt-cells属性指定。

搜寻interrupt mapping 表,匹配子组件的unit-address/interrupt specifier 对,由于一些interrupt specifier 位是无关的,可以通过interrupt-map-mask进行and位掩码来忽略掉不匹配。

子节点和父节点都需要#address-cells #interruptcells 属性。如果unit address不需要,则必须显示的配置#address-cells 0.

interrupt-map-mask

属性名: interrupt-map-mask

值类型: 编码为位掩码bit mask

中断树中的连接Nexus节点,其 interrupt-map-mask属性指定了一个掩码,该掩码与在interrupt-map属性中指定的表中查找的传入单元中断说明符interrupt specifier 进行AND运算

#interrupt-cells

属性名: #interrupt-cells

值类型:

定义了对中断域interrupt domain的中断说明符interrupt specifier进行编码所需的单元数。

Interrupt Mapping示例

soc {compatible = "simple-bus";#address-cells = <1>;#size-cells = <1>;
open-pic {clock-frequency = <0>;interrupt-controller;#address-cells = <0>;#interrupt-cells = <2>;};
pci {#interrupt-cells = <1>;#size-cells = <2>;#address-cells = <3>;interrupt-map-mask = <0xf800 0 0 7>;
interrupt-map = </* IDSEL 0x11 - PCI slot 1 */0x8800 0 0 1 &open-pic 2 1 /* INTA */0x8800 0 0 2 &open-pic 3 1 /* INTB */0x8800 0 0 3 &open-pic 4 1 /* INTC */0x8800 0 0 4 &open-pic 1 1 /* INTD *//* IDSEL 0x12 - PCI slot 2 */0x9000 0 0 1 &open-pic 3 1 /* INTA */0x9000 0 0 2 &open-pic 4 1 /* INTB */0x9000 0 0 3 &open-pic 1 1 /* INTC */0x9000 0 0 4 &open-pic 2 1 /* INTD */>;};}

open-picinterrupt-controller属性说明该节点是中断控制器

0x8800 0 0 1 &open-pic 2 1 /* INTA */为例子

对应五元组

child unit address=0x8800 0 0

子节点是INTA所在的总显示pci,而pci#address-cells = <3>;,所以child unit address占用332位单元。

PCI地址0x8800 0 0解析为

bus number (0x0 << 16), device number (0x11 << 11), function number(0x0 << 8).

对应的0x9000 0 0解析为

bus number (0x0 << 16), device number (0x12 << 11),  function number(0x0 << 8).

child interrupt specifier = 1

interrupt-map所在节点即pci节点的#interrupt-cells = <1>;,所以child interrupt specifier占用132位单元。

interrupt-parent = &open-pic

表示映射到父中断open-pic

parent unit address= empty

interrupt-parent = &open-pic指向的节点,即open-pic#address-cells = <0>;

所以parent unit addressempty

parent interrupt specifier=2 1

interrupt-parent = &open-pic指向的节点,即open-pic#interrupt-cells = <2>;

所以parent interrupt specifier占用两个32位单元。

<2 1>Open PIC中断控制器定义。值<2>指定INTA连接到的中断控制器上的中断源号。值<1>指定电平编码。

这样中断控制器的中断号2就代表了INTA

interrupt-map-mask = <0xf800 0 0 7> 表示

在中断映射表中执行查找之前,将此掩码应用于子单元中断说明符。

如果要查找IDSEL 0x12(插槽2,功能号0x3INTB(interrupt specifier=2)open pic中断源号,将执行以下步骤

child unit address  interrupt specifier组成值<0x9300 0 0 2>

其中

child unit address = bus number (0x0 << 16), device number (0x12 << 11), and function

number (0x3 << 8).

interrupt specifier-2,表示INTB

<0xf800 0 0 7><0x9300 0 0 2> and得到<0x9000 0 0 2>

在中断映射表中查找该结果,该表映射到父中断说明符<4 1>

即对应

0x9000 0 0 2 &open-pic 4 1 /* INTB */

这一行。

Nexus节点和规范映射

Nexus节点属性

nexus节点应具有#-cells 属性,其中是一些说明符空间,‘gpio’, ‘clock’, ‘reset’ 等。

-map

属性名: -map

属性值: 编码为任意数量的说明符映射条目

该属性将一个说明符域与一组父说明符域连接起来,并描述子域中的说明符如何映射到各自的父域。

映射是一个表,其中每一行都是一个映射条目,由三个组件组成:child specifier子说明符, specifier parent说明符, parent specifier父说明符。

child specifier:

映射的子节点的说明符,其值需要多少个32位单元,由-map 所在连接节点的#-cells属性指定。

specifier parent:

说明符的父,一个值,指向子域映射到的说明符父级。

parent specifier:

父域中的说明符。

其值需要多少个32位单元,由specifier parent#-cells属性决定。

通过将说明符与映射中的子说明符进行匹配,在映射表上执行查找。由于说明符中的某些字段可能不相关或需要修改,因此在执行查找之前会应用掩码。此掩码在-map-mask属性中定义。

同样,当映射说明符时,单元说明符中的某些字段可能需要保持不变,并从子节点传递到父节点。此时可以指定 -map-pass-thru 属性。可以指定将掩码应用于子说明符,并复制与父单元说明符匹配的任何位。

-map-mask

属性值: -map-mask

值类型: 编码为位掩码bit mask

可以在nexus节点中指定。此属性指定了一个掩码,该掩码与在-map 属性中指定的表中查找的子单元说明符进行AND运算。如果未指定此属性,则假定掩码是设置了所有位的掩码。

-map-pass-thru

属性名: -map-pass-thru

值类型: 编码为位掩码 bit mask

可以在nexus节点中指定。

此属性指定了一个掩码,该掩码应用于在-map属性中指定的表中查找的子单元说明符。子单位说明符中的任何匹配位都会复制到父说明符。如果未指定此属性,则假定掩码为未设置位的掩码。

#-cells

属性名: #-cells

值类型:

#-cells属性定义了对域的说明符进行编码所需的单元数。

specifier映射示例

下面显示了具有两个GPIO控制器的设备树片段的表示,以及一个specifier map 示例,用于描述两个控制器上的几个gpios通过板上的连接器到设备的GPIO路由。扩展设备节点位于连接器节点的一侧,带有两个GPIO控制器的SoC位于连接器的另一侧。

soc {soc_gpio1: gpio-controller1 {#gpio-cells = <2>;};
soc_gpio2: gpio-controller2 {#gpio-cells = <2>;};};
connector: connector {#gpio-cells = <2>;gpio-map = <0 0 &soc_gpio1 1 0>,<1 0 &soc_gpio2 4 0>,<2 0 &soc_gpio1 3 0>,<3 0 &soc_gpio2 2 0>;gpio-map-mask = <0xf 0x0>;gpio-map-pass-thru = <0x0 0x1>;};
expansion_device {reset-gpios = <&connector 2 GPIO_ACTIVE_LOW>;};

gpio-map每一行由3元组组成

<0 0 &soc_gpio1 1 0>,为例

child specifier: 0 0

-map 所在连接节点connector#gpio-cells = <2>;

所以child specifier占用232位单元

specifier parent: &soc_gpio1

指向soc_gpio1

parent specifier: 1 0

specifier parent指向的节点soc_gpio1#gpio-cells = <2>;

所以parent specifier占用232位单元

<1 0>的含义由GPIO控制器决定,比如1可以表示引脚号,0表示高有效还是低有效。

其中

gpio-map-mask = <0xf 0x0>;

假设要从expansion_devicereset-gpios ,查找对应connectorGPIO2specifier源号,

specifier<2 GPIO_ACTIVE_LOW><2 0>

gpio-map-mask = <0xf 0x0><2 0> AND

得到<2 0>

查找

gpio-map,找到<2 0 &soc_gpio1 3 0>,

parent &soc_gpio1 phandleparent specifier= <3 0>

gpio-map-pass-thru = parent specifier<0x0 0x1>取反,和<3 0> AND得到<3 0>  (即将对应的位清零)

gpio-map-pass-thru child specifier<2 0> 得到<0 0>

然后<3 0> <0 0>或, 得到<3 0>&soc_gpio1得到 <&soc_gpio1

3 GPIO_ACTIVE_LOW>.

DTB数据格式

.dtb格式文件对应如下

其中free space可能没有,主要是用于填充对齐

头信息

对应如下,所有数据都是大端,32

struct fdt_header {uint32_t magic;uint32_t totalsize;uint32_t off_dt_struct;uint32_t off_dt_strings;uint32_t off_mem_rsvmap;uint32_t version;uint32_t last_comp_version;uint32_t boot_cpuid_phys;uint32_t size_dt_strings;uint32_t size_dt_struct;};

magic:

固定为0xd00dfeed

totalsize:

此字段应包含设备树数据结构的总大小(以字节为单位)。该大小应包括结构的所有部分:标头、内存保留块、结构块和字符串块,以及块之间或最终块之后的任何自由空间间隙.

off_dt_struct:

表示structure block位于从最开始的偏移字节数。

off_dt_strings:

表示strings block位于从最开始的偏移字节数。

off_mem_rsvmap:

表示memory reservation block位于从最开始的偏移字节数。

Version:

此字段应包含设备树数据结构的版本。比如17

last_comp_version:

此字段应包含设备树数据结构的最低版本,所使用的版本与该版本向后兼容。

17版本最早只能支持16,所以Version17时这里只能是16.

boot_cpuid_phys:

此字段应包含系统引导CPU的物理ID。它应与设备树中该CPU节点的reg属性中给出的物理ID相同

size_dt_strings

表示strings block section 的有效字节数。

size_dt_struct

表示 structure block 的有效字节数。

Memory Reservation Block

内存保留块向客户端程序提供物理内存中保留的区域列表;也就是说,它不应用于一般的内存分配。它用于保护重要数据结构不被客户端程序覆盖。例如,在某些具有IOMMU的系统上,需要以这种方式保护由DTSpec引导程序初始化的TCE(转换控制条目)表。同样,在客户端程序运行时使用的任何引导程序代码或数据都需要保留(例如,Open Firmware平台上的RTAS)。DTSpec不要求引导程序提供任何此类运行时组件,但它并不禁止实现作为扩展这样做

更具体地说,客户端程序不应访问保留区域中的内存,除非引导程序提供的其他信息明确指示它应该这样做。然后,客户端程序可以以指示的方式访问保留内存的指示部分。引导程序可以向客户端程序指示保留内存的特定用途的方法可能出现在本文档、其可选扩展或特定于平台的文档中。

引导程序提供的保留区域可能(但并非必须)包含设备树blob本身。客户端程序应确保在使用之前不会覆盖此数据结构,无论它是否在保留区域。

必须保留在内存节点中声明的任何内存,这些内存由引导程序访问,或者在客户端进入后被引导程序访问。这种类型的访问的示例包括(例如,通过非保护的虚拟页面进行推测性内存读取)。

这一要求是必要的,因为任何未保留的内存都可能被具有任意存储属性的客户端程序访问。

由引导程序或由引导程序引起的对保留内存的任何访问都必须按照不禁止缓存和不要求内存一致性的方式进行(即WIMG=0bx01x),此外,对于Book III-S实现,也不要求直写(即WIMG=0b001x)。此外,如果支持VLE存储属性,则对保留内存的所有访问都必须在VLE=0时完成

此要求是必要的,因为允许客户端程序映射具有存储属性的内存,这些存储属性指定为不需要直写、不禁止缓存、需要内存一致性(即WIMG=0b001x),在支持的情况下VLE=0。客户端程序可以使用包含保留内存的大型虚拟页面。但是,客户端程序可能不会修改保留内存,因此引导程序可能会在架构上允许此存储属性的冲突值的情况下,以需要直写的方式执行对保留内存的访问。

内存预留块由一对64位大端序整数组成,每对整数由以下C结构表示

struct fdt_reserve_entry {

uint64_t address;

uint64_t size;

}

每对都给出了保留内存区域的物理地址和字节大小。这些给定区域不得相互重叠。保留块列表应以地址和大小均等于0的条目终止。请注意,地址和大小值始终为64位。在32CPU上,值的前32位将被忽略。

内存预留块中的每个uint64_t,以及整个内存预留块,应位于设备树blob开头的8字节对齐偏移处。

/reserved-memory 节点(一样,当通过[UEFI]启动时,内存保留块中的条目也必须列在通过GetMemoryMap获得的系统内存映射中,以防止UEFI应用程序的分配。内存保留块条目应按EfiReservedMemoryType类型列出。

Structure Block

结构块描述了设备树本身的结构和内容。它由一系列带有数据的令牌组成,如下所述。这些被组织成线性树结构,

结构块中的每个令牌,以及结构块本身,应位于与设备树blob开头对齐的4字节偏移处

词汇结构Lexical structure

结构块由一系列片段组成,每个片段都以一个标记开始,即一个大端32位整数。一些令牌后面是额外的数据,其格式由令牌值决定。所有令牌应在32位边界上对齐,这可能需要在前一个令牌的数据后插入填充字节(值为0x0)。

五种令牌类型如下:

FDT_BEGIN_NODE (0x00000001):

标记节点的开始。它后面应跟节点的单元名称作为额外数据。名称存储为以空结尾的字符串,如果有,应包括单元地址。如果对齐需要,节点名称后面是零填充字节,然后是下一个令牌,可以是除FDT_END之外的任何令牌。

FDT_END_NODE (0x00000002):

标志节点的结束。此令牌没有额外数据;因此,紧随其后的是下一个令牌,它可以是除FDT_PROP之外的任何令牌。

FDT_PROP (0x00000003)

标记设备树中一个属性的开始。随后应提供描述该属性的额外数据。此数据首先由属性的长度和名称组成,表示为以下C结构:

struct {

uint32_t len;

uint32_t nameoff;

}

此结构中的两个字段都是32位大端序整数。

len以字节为单位给出属性值的长度(可能为零,表示属性为空)。

nameoff在字符串块中给出了一个偏移量,在该块中,属性的名称被存储为以null结尾的字符串。在此结构之后,属性的值以长度为len的字节字符串给出。此值后面是零填充字节(如果需要),以对齐下一个32位边界,然后是下一个令牌,可以是除FDT_END之外的任何令牌。

FDT_NOP (0x00000004)

解析设备树的任何程序都将忽略FDT_NO令牌。此令牌没有额外数据;因此,紧随其后的是下一个令牌,可以是任何有效的令牌。树中的属性或节点定义可以用FDT_NOP标记覆盖,以将其从树中删除,而无需在设备树blob中移动树表示的其他部分.

FDT_END (0x00000009)

标记结构块的结束。应该只有一个FDT_END令牌,并且它应该是结构块中的最后一个令牌。它没有额外的数据;因此,紧随FDT_END令牌之后的字节从结构块的开头偏移,等于设备树blob标头中size_dt_struct字段的值。

Tree structure

设备树结构表示为线性树:每个节点的表示以FDT_BEGIN_node令牌开始,以FDT_END_node令牌结束。节点的属性和子节点(如果有的话)在FDT_END_node之前表示,因此这些子节点的FDT_BEGIN_nodeFDT_END_node令牌嵌套在父节点的令牌中

结构块作为一个整体由根节点的表示(包含所有其他节点的表示)组成,后面是一个FDT_END令牌,用于标记整个结构块的结束。

更确切地说,每个节点的表示由以下组件组成:

l(可选)任意数量的FDT_NO令牌

lFDT_BEGIN_NODE

null结尾的字符串表示的节点名称

[零填充字节以对齐4字节边界]

l对于节点的每个属性:

(可选)任意数量的FDT_NO令牌

FDT_PROP

属性信息

[零填充字节以对齐4字节边界]

l此格式中所有子节点的表示

l(可选)任意数量的FDT_NO令牌

lFDT_END_NODE

注意,此过程要求特定节点的所有属性定义都位于该节点的任何子节点定义之前。虽然如果属性和子节点混合在一起,结构就不会含糊不清,但处理扁平树所需的代码通过这一要求得到了简化

Strings Block

字符串块包含表示树中使用的所有属性名称的字符串。这些以空结尾的字符串在本节中简单地连接在一起,并通过偏移量从Structure Block引用到Strings Block

Strings Block没有对齐约束,可能出现在设备树blob开头的任何偏移处

对齐要求

设备树blob应位于8位对齐的地址。为了保持32位机器的向后兼容性,一些软件支持4字节对齐,但这不符合DTSpec标准

对于在没有未对齐内存访问的情况下使用的内存预留和结构块中的数据,它们应位于适当对齐的内存地址。具体来说,Memory Reservation Block应与8字节边界对齐,Structure Block应与4字节边界对齐。

此外,设备树blob作为一个整体可以重新定位,而不会破坏子块的对齐

Structure BlockString Block应与设备树blob的开头对齐偏移。为了确保块在内存中的对齐,只需确保设备树作为一个整体被加载到与任何子块的最大对齐对齐的地址,即8字节边界对齐即可。符合DTSpec的引导程序应在将设备树blob传递给客户端程序之前,在对齐的地址加载设备树blob。如果DTSpec客户端程序在内存中重新定位设备树blob,它应该只将其重新定位到另一个8字节对齐的地址.

DTS文件

设备树源(DTS)格式是设备树的文本表示形式,dtc可以将其处理为内核所需形式的二进制设备树。以下描述不是DTS的正式语法定义,但描述了用于表示设备树的基本构造。

DTS文件的名称应以“.DTS”结尾。

编译指令

DTS文件中可以包含其他源文件。包含文件的名称应以“.dtsi”结尾。包含的文件可以反过来包含其他文件。

/include/ "FILE"

标签

源格式允许将标签附加到设备树中的任何节点或属性值。Phandle和路径引用可以通过引用标签自动生成,而不是显式指定Phandle值或节点的完整路径。标签仅用于设备树源格式,不编码为DTB二进制文件

标签的长度应在131个字符之间,仅由表中的字符组成,不得以数字开头。

标签是通过在标签名称后附加冒号()创建的。通过在标签名称前加上与号(“&”)来创建引用。

节点和属性定义

设备树节点由节点名称和单元地址定义,大括号标记节点定义的开始和结束。它们前面可能有一个标签。

[label:] node-name[@unit-address] {

[properties definitions]

[child nodes]

};

节点可能包含属性定义和/或子节点定义。如果两者都存在,则属性应位于子节点之前。

可以删除以前定义的节点

/delete-node/ node-name;

/delete-node/ &label;

属性定义是以下形式的名称-值对:

[label:] property-name = value;

除了具有空(零长度)值的属性,其形式如下:

[label:] property-name;

可以删除以前定义的属性。

/delete-property/ property-name; 

属性值可以定义为32位整数单元的数组、以空结尾的字符串、字节串或它们的组合。

单元格数组由尖括号表示,尖括号围绕着一个空格分隔的C风格整数列表。例子:

interrupts = <17 0xc>; 

值可以表示为括号内的算术、位或逻辑表达式。

Arithmetic operators

+ add

- subtract

* multiply

/ divide

% modulo 

Bitwise operators

& and

| or

^ exclusive or

~ not

<< left shift

>> right shift

Logical operators

&& and

|| or

! Not

Relational operators

< less than

> greater than

<= less than or equal

>= greater than or equal

== equal

!= not equal

Ternary operators

?: (condition ? value_if_true : value_if_false)

64位值由两个32位单元表示。例子:

clock-frequency = <0x00000001 0x00000000>; 

以空结尾的字符串值使用双引号表示(属性值被认为包含终止的null字符)。例子:

compatible = "simple-bus"; 

字节串被括在方括号[]中,每个字节由两个十六进制数字表示。每个字节之间的空格是可选的。例子:

local-mac-address = [00 00 12 34 56 78]; 

等效于

local-mac-address = [000012345678]; 

值可能有几个逗号分隔的组件,这些组件连接在一起。例子:

compatible = "ns16550", "ns8250";

example = <0xf00f0000 19>, "a strange property format"; 

在单元数组中,对另一个节点的引用将扩展到该节点的phandle。引用后面可能跟有节点的标签。例子:

interrupt-parent = < &mpic >; 

或者,它们后面可能是大括号中节点的完整路径。例子

interrupt-parent = < &{/soc/interrupt-controller@40000} >; 

在单元数组外部,对另一个节点的引用将扩展到该节点的完整路径。例子:

ethernet0 = &EMAC0; 

标签也可能出现在属性值的任何组成部分之前或之后,或者出现在单元格数组的单元格之间,或者在字节串的字节之间。示例

reg = reglabel: <0 sizelabel: 0x1000000>;

prop = [ab cd ef byte4: 00 ff fe];

str = start: "string value" end: ;

文件布局

dts-v1/;[memory reservations]/ {[property definitions][child nodes]};

/dts-v1/;应存在以将文件标识为版本1 DTS(没有此标签的DTS文件将被dtc视为过时的版本0,除了其他小但不兼容的更改外,该版本对整数使用不同的格式)。

内存预留由以下表格中的线条表示:

/memreserve/

; 

其中

是64位C风格整数。

/* Reserve memory region 0x10000000..0x10003fff */

/memreserve/ 0x10000000 0x4000;

/{…};部分定义了设备树的根节点,所有设备树数据都包含在其中。

支持C风格(/*…\*/)和C++风格(//)注释。

使用DTC工具

https://github.com/dgibson/dtc项目

安装dtc工具

sudo apt-get install device-tree-compiler

dumpqemu的设备树

qemu-system-riscv64 -M virt,dumpdtb=dump_qemu.dtb

编译设备树dts->dtb

dtc -o dump_qemu.dtb -I dts -O dtb dump_qemu.dts

反编译设备树dtb->dts

dtc -o dump_qemu.dts -I dtb -O dts dump_qemu.dtb

设备库解析代码

https://github.com/dgibson/dtc项目

Opensbi中使用的fdt解析源码位于lib/utils/fdt/

Linux中相关代码位于

drivers/of/fdt.c

总结

分两篇分享了设备树的基本内容,包括设备树的语法,dtbdts文件,为后面调试打下了基础。在opensbilinux kernel调试时遇到设备树相关的解析的地方就会有比较清晰的思路,能够调试分析设备树相关的解析处理是否正确,以及相应的设备树传递参数信息等的处理流程。



评论
  • 嘿,咱来聊聊RISC-V MCU技术哈。 这RISC-V MCU技术呢,简单来说就是基于一个叫RISC-V的指令集架构做出的微控制器技术。RISC-V这个啊,2010年的时候,是加州大学伯克利分校的研究团队弄出来的,目的就是想搞个新的、开放的指令集架构,能跟上现代计算的需要。到了2015年,专门成立了个RISC-V基金会,让这个架构更标准,也更好地推广开了。这几年啊,这个RISC-V的生态系统发展得可快了,好多公司和机构都加入了RISC-V International,还推出了不少RISC-V
    丙丁先生 2025-01-21 12:10 111浏览
  • 2024年是很平淡的一年,能保住饭碗就是万幸了,公司业绩不好,跳槽又不敢跳,还有一个原因就是老板对我们这些员工还是很好的,碍于人情也不能在公司困难时去雪上加霜。在工作其间遇到的大问题没有,小问题还是有不少,这里就举一两个来说一下。第一个就是,先看下下面的这个封装,你能猜出它的引脚间距是多少吗?这种排线座比较常规的是0.6mm间距(即排线是0.3mm间距)的,而这个规格也是我们用得最多的,所以我们按惯性思维来看的话,就会认为这个座子就是0.6mm间距的,这样往往就不会去细看规格书了,所以这次的运气
    wuliangu 2025-01-21 00:15 183浏览
  •  万万没想到!科幻电影中的人形机器人,正在一步步走进我们人类的日常生活中来了。1月17日,乐聚将第100台全尺寸人形机器人交付北汽越野车,再次吹响了人形机器人疯狂进厂打工的号角。无独有尔,银河通用机器人作为一家成立不到两年时间的创业公司,在短短一年多时间内推出革命性的第一代产品Galbot G1,这是一款轮式、双臂、身体可折叠的人形机器人,得到了美团战投、经纬创投、IDG资本等众多投资方的认可。作为一家成立仅仅只有两年多时间的企业,智元机器人也把机器人从梦想带进了现实。2024年8月1
    刘旷 2025-01-21 11:15 390浏览
  • 随着消费者对汽车驾乘体验的要求不断攀升,汽车照明系统作为确保道路安全、提升驾驶体验以及实现车辆与环境交互的重要组成,日益受到业界的高度重视。近日,2024 DVN(上海)国际汽车照明研讨会圆满落幕。作为照明与传感创新的全球领导者,艾迈斯欧司朗受邀参与主题演讲,并现场展示了其多项前沿技术。本届研讨会汇聚来自全球各地400余名汽车、照明、光源及Tier 2供应商的专业人士及专家共聚一堂。在研讨会第一环节中,艾迈斯欧司朗系统解决方案工程副总裁 Joachim Reill以深厚的专业素养,主持该环节多位
    艾迈斯欧司朗 2025-01-16 20:51 195浏览
  •  光伏及击穿,都可视之为 复合的逆过程,但是,复合、光伏与击穿,不单是进程的方向相反,偏置状态也不一样,复合的工况,是正偏,光伏是零偏,击穿与漂移则是反偏,光伏的能源是外来的,而击穿消耗的是结区自身和电源的能量,漂移的载流子是 客席载流子,须借外延层才能引入,客席载流子 不受反偏PN结的空乏区阻碍,能漂不能漂,只取决于反偏PN结是否处于外延层的「射程」范围,而穿通的成因,则是因耗尽层的过度扩张,致使跟 端子、外延层或其他空乏区 碰触,当耗尽层融通,耐压 (反向阻断能力) 即告彻底丧失,
    MrCU204 2025-01-17 11:30 182浏览
  • 临近春节,各方社交及应酬也变得多起来了,甚至一月份就排满了各式约见。有的是关系好的专业朋友的周末“恳谈会”,基本是关于2025年经济预判的话题,以及如何稳定工作等话题;但更多的预约是来自几个客户老板及副总裁们的见面,他们为今年的经济预判与企业发展焦虑而来。在聊天过程中,我发现今年的聊天有个很有意思的“点”,挺多人尤其关心我到底是怎么成长成现在的多领域风格的,还能掌握一些经济趋势的分析能力,到底学过哪些专业、在企业管过哪些具体事情?单单就这个一个月内,我就重复了数次“为什么”,再辅以我上次写的:《
    牛言喵语 2025-01-22 17:10 41浏览
  • 高速先生成员--黄刚这不马上就要过年了嘛,高速先生就不打算给大家上难度了,整一篇简单但很实用的文章给大伙瞧瞧好了。相信这个标题一出来,尤其对于PCB设计工程师来说,心就立马凉了半截。他们辛辛苦苦进行PCB的过孔设计,高速先生居然说设计多大的过孔他们不关心!另外估计这时候就跳出很多“挑刺”的粉丝了哈,因为翻看很多以往的文章,高速先生都表达了过孔孔径对高速性能的影响是很大的哦!咋滴,今天居然说孔径不关心了?别,别急哈,听高速先生在这篇文章中娓娓道来。首先还是要对各位设计工程师的设计表示肯定,毕竟像我
    一博科技 2025-01-21 16:17 100浏览
  • 现在为止,我们已经完成了Purple Pi OH主板的串口调试和部分配件的连接,接下来,让我们趁热打铁,完成剩余配件的连接!注:配件连接前请断开主板所有供电,避免敏感电路损坏!1.1 耳机接口主板有一路OTMP 标准四节耳机座J6,具备进行音频输出及录音功能,接入耳机后声音将优先从耳机输出,如下图所示:1.21.2 相机接口MIPI CSI 接口如上图所示,支持OV5648 和OV8858 摄像头模组。接入摄像头模组后,使用系统相机软件打开相机拍照和录像,如下图所示:1.3 以太网接口主板有一路
    Industio_触觉智能 2025-01-20 11:04 150浏览
  •     IPC-2581是基于ODB++标准、结合PCB行业特点而指定的PCB加工文件规范。    IPC-2581旨在替代CAM350格式,成为PCB加工行业的新的工业规范。    有一些免费软件,可以查看(不可修改)IPC-2581数据文件。这些软件典型用途是工艺校核。    1. Vu2581        出品:Downstream     
    电子知识打边炉 2025-01-22 11:12 49浏览
  • 80,000人到访的国际大展上,艾迈斯欧司朗有哪些亮点?感未来,光无限。近日,在慕尼黑electronica 2024现场,ams OSRAM通过多款创新DEMO展示,以及数场前瞻洞察分享,全面展示自身融合传感器、发射器及集成电路技术,精准捕捉并呈现环境信息的卓越能力。同时,ams OSRAM通过展会期间与客户、用户等行业人士,以及媒体朋友的深度交流,向业界传达其以光电技术为笔、以创新为墨,书写智能未来的深度思考。electronica 2024electronica 2024构建了一个高度国际
    艾迈斯欧司朗 2025-01-16 20:45 423浏览
  • Ubuntu20.04默认情况下为root账号自动登录,本文介绍如何取消root账号自动登录,改为通过输入账号密码登录,使用触觉智能EVB3568鸿蒙开发板演示,搭载瑞芯微RK3568,四核A55处理器,主频2.0Ghz,1T算力NPU;支持OpenHarmony5.0及Linux、Android等操作系统,接口丰富,开发评估快人一步!添加新账号1、使用adduser命令来添加新用户,用户名以industio为例,系统会提示设置密码以及其他信息,您可以根据需要填写或跳过,命令如下:root@id
    Industio_触觉智能 2025-01-17 14:14 121浏览
  • 数字隔离芯片是一种实现电气隔离功能的集成电路,在工业自动化、汽车电子、光伏储能与电力通信等领域的电气系统中发挥着至关重要的作用。其不仅可令高、低压系统之间相互独立,提高低压系统的抗干扰能力,同时还可确保高、低压系统之间的安全交互,使系统稳定工作,并避免操作者遭受来自高压系统的电击伤害。典型数字隔离芯片的简化原理图值得一提的是,数字隔离芯片历经多年发展,其应用范围已十分广泛,凡涉及到在高、低压系统之间进行信号传输的场景中基本都需要应用到此种芯片。那么,电气工程师在进行电路设计时到底该如何评估选择一
    华普微HOPERF 2025-01-20 16:50 73浏览
  • 本文介绍瑞芯微开发板/主板Android配置APK默认开启性能模式方法,开启性能模式后,APK的CPU使用优先级会有所提高。触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。源码修改修改源码根目录下文件device/rockchip/rk3562/package_performance.xml并添加以下内容,注意"+"号为添加内容,"com.tencent.mm"为AP
    Industio_触觉智能 2025-01-17 14:09 164浏览
  • 日前,商务部等部门办公厅印发《手机、平板、智能手表(手环)购新补贴实施方案》明确,个人消费者购买手机、平板、智能手表(手环)3类数码产品(单件销售价格不超过6000元),可享受购新补贴。每人每类可补贴1件,每件补贴比例为减去生产、流通环节及移动运营商所有优惠后最终销售价格的15%,每件最高不超过500元。目前,京东已经做好了承接手机、平板等数码产品国补优惠的落地准备工作,未来随着各省市关于手机、平板等品类的国补开启,京东将第一时间率先上线,满足消费者的换新升级需求。为保障国补的真实有效发放,基于
    华尔街科技眼 2025-01-17 10:44 221浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦