内核启动时的中断和时钟的初始化流程

总结

内核通过start_kernel从汇编进入c世界,在用serup_arch设置体系架构的时候,会返回当前机器的machine_desc;然后init_IRQ和time_init来初始化中断。时钟子系统

start_kernelsetup_arch(&command_line);mdesc = setup_machine_fdt(atags_vaddr) //返回成功匹配的machine_descmachine_desc = mdesc; //赋值给全局变量machine_desc,以供后续初始化等操作使用,比如给下面的中断和时钟子系统使用......init_IRQmachine_desc->init_irq()irqchip_initof_irq_init(__irqchip_of_table) //IRQCHIP_DECLARE(gic_v3, "arm,gic-v3", gic_of_init)......time_initif (machine_desc->init_time)machine_desc->init_time of_clk_init(NULL);timer_probe();elseof_clk_init(NULL);for_each_matching_node_and_match(np, matches, &match) // CLK_OF_DECLARE(asr1803_clk, "asr,asr1803-clock", asr1803_clk_init);由晶振+pll给cpu核心和外设产生时钟timer_probe();for_each_matching_node_and_match(np, __timer_of_table, &match) //TIMER_OF_DECLARE(mmp_timer, "mrvl,mmp-timer", mmp_dt_init_timer);定时器外设

setup_arch

setup_machine_fdt

of_flat_dt_match_machine去匹配设备树,和DT_MACHINE_START定义的machine_desc中的兼容属性,并返回这个machine_desc

const struct machine_desc * __init setup_machine_fdt(void *dt_virt)
{const struct machine_desc *mdesc, *mdesc_best = NULL;#if defined(CONFIG_ARCH_MULTIPLATFORM) || defined(CONFIG_ARM_SINGLE_ARMV7M)DT_MACHINE_START(GENERIC_DT, "Generic DT based system").l2c_aux_val = 0x0,.l2c_aux_mask = ~0x0,MACHINE_ENDmdesc_best = &__mach_desc_GENERIC_DT;
#endifif (!dt_virt || !early_init_dt_verify(dt_virt))return NULL;mdesc = of_flat_dt_match_machine(mdesc_best, arch_get_next_mach);if (!mdesc) {const char *prop;int size;unsigned long dt_root;early_print("\nError: unrecognized/unsupported ""device tree compatible list:\n[ ");dt_root = of_get_flat_dt_root();prop = of_get_flat_dt_prop(dt_root, "compatible", &size);while (size > 0) {early_print("'%s' ", prop);size -= strlen(prop) + 1;prop += strlen(prop) + 1;}early_print("]\n\n");dump_machine_table(); /* does not return */}/* We really don't want to do this, but sometimes firmware provides buggy data */if (mdesc->dt_fixup)mdesc->dt_fixup();early_init_dt_scan_nodes();/* Change machine number to match the mdesc we're using */__machine_arch_type = mdesc->nr;return mdesc;
}

arch_get_next_mach

__arch_info_begin就是在链接脚本vmlinux.lds.S定义

static const void * __init arch_get_next_mach(const char *const **match)
{static const struct machine_desc *mdesc = __arch_info_begin;const struct machine_desc *m = mdesc;if (m >= __arch_info_end)return NULL;mdesc++;*match = m->dt_compat;return m;
}//arch/arm/kernel/vmlinux.lds.S.init.arch.info : {__arch_info_begin = .;*(.arch.info.init)__arch_info_end = .;}
......
ASSERT((__proc_info_end - __proc_info_begin), "missing CPU support")
ASSERT((__arch_info_end - __arch_info_begin), "no machine record defined")

DT_MACHINE_START

arch/arm/include/asm/mach/arch.h的DT_MACHINE_START将machine_desc放进__arch_info_begin -- __arch_info_end的段空间中

extern const struct machine_desc __arch_info_begin[], __arch_info_end[];
#define for_each_machine_desc(p)			\for (p = __arch_info_begin; p < __arch_info_end; p++)/** Set of macros to define architecture features.  This is built into* a table by the linker.*/
#define MACHINE_START(_type,_name)			\
static const struct machine_desc __mach_desc_##_type	\__used							\__attribute__((__section__(".arch.info.init"))) = {	\.nr		= MACH_TYPE_##_type,		\.name		= _name,#define MACHINE_END				\
};#define DT_MACHINE_START(_name, _namestr)		\
static const struct machine_desc __mach_desc_##_name	\__used							\__attribute__((__section__(".arch.info.init"))) = {	\.nr		= ~0,				\.name		= _namestr,#endif

dt_compat

ARM Linux 3.x在引入设备树之后,MACHINE_START变更为DT_MACHINE_START,其中含有一个.dt_compat成员,用于表明相关的设备与.dts中根节点的兼容属性兼容关系

/ {model = "ASR 1828(C260SA) CPE";compatible = "asr,1803-evb", "asr,1803";.......
}static const char *asr1803_dt_board_compat[] __initdata = {"asr,1803-evb",NULL,
};
DT_MACHINE_START(ASR1803_DT, "ASR ASR1803 (Device Tree Support)").map_io         = mmp_map_io,.init_irq       = irqchip_init,.init_time      = mmp_init_time,.reserve        = asr1803_reserve,.init_machine   = asr1803_dt_init_machine,.dt_compat      = asr1803_dt_board_compat,.restart        = mmp_arch_restart,
MACHINE_END

init_IRQ

init_IRQ会调用machine_desc->init_irq,根据上面DT_MACHINE_START的定义,也即irqchip_init;中断控制器驱动会定义IRQCHIP_DECLARE(gic_v3, "arm,gic-v3", gicv3_of_init);经过_OF_DECLARE展开,就会产生__irqchip_of_table;因为还没初始化deivce-bus-driver这套去走match/probe自动回调驱动的probe,又要兼容设备树,只能前期去主动匹配后,调用初始化函数

#define _OF_DECLARE(table, name, compat, fn, fn_type)			\static const struct of_device_id __of_table_##name		\__used __section(__##table##_of_table)			\__aligned(__alignof__(struct of_device_id))		\= { .compatible = compat,				\.data = (fn == (fn_type)NULL) ? fn : fn  }void __init irqchip_init(void)
{of_irq_init(__irqchip_of_table);acpi_probe_device_table(irqchip);
}

然后通过for_each_matching_node_and_match 来解析设备树节点,并在后续主动调用.data,也即gicv3_of_init。主要代码为desc->irq_init_cb = match->data; desc->irq_init_cb(desc->dev,desc->interrupt_parent);

void __init of_irq_init(const struct of_device_id *matches)
{const struct of_device_id *match;struct device_node *np, *parent = NULL;struct of_intc_desc *desc, *temp_desc;struct list_head intc_desc_list, intc_parent_list;INIT_LIST_HEAD(&intc_desc_list);INIT_LIST_HEAD(&intc_parent_list);for_each_matching_node_and_match(np, matches, &match) {if (!of_property_read_bool(np, "interrupt-controller") ||!of_device_is_available(np))continue;if (WARN(!match->data, "of_irq_init: no init function for %s\n",match->compatible))continue;/** Here, we allocate and populate an of_intc_desc with the node* pointer, interrupt-parent device_node etc.*/desc = kzalloc(sizeof(*desc), GFP_KERNEL);if (!desc) {of_node_put(np);goto err;}desc->irq_init_cb = match->data;desc->dev = of_node_get(np);desc->interrupt_parent = of_irq_find_parent(np);if (desc->interrupt_parent == np)desc->interrupt_parent = NULL;list_add_tail(&desc->list, &intc_desc_list);}/** The root irq controller is the one without an interrupt-parent.* That one goes first, followed by the controllers that reference it,* followed by the ones that reference the 2nd level controllers, etc.*/while (!list_empty(&intc_desc_list)) {/** Process all controllers with the current 'parent'.* First pass will be looking for NULL as the parent.* The assumption is that NULL parent means a root controller.*/list_for_each_entry_safe(desc, temp_desc, &intc_desc_list, list) {int ret;if (desc->interrupt_parent != parent)continue;list_del(&desc->list);of_node_set_flag(desc->dev, OF_POPULATED);pr_debug("of_irq_init: init %pOF (%p), parent %p\n",desc->dev,desc->dev, desc->interrupt_parent);ret = desc->irq_init_cb(desc->dev,desc->interrupt_parent);if (ret) {of_node_clear_flag(desc->dev, OF_POPULATED);kfree(desc);continue;}/** This one is now set up; add it to the parent list so* its children can get processed in a subsequent pass.*/list_add_tail(&desc->list, &intc_parent_list);}/* Get the next pending parent that might have children */desc = list_first_entry_or_null(&intc_parent_list,typeof(*desc), list);if (!desc) {pr_err("of_irq_init: children remain, but no parents\n");break;}list_del(&desc->list);parent = desc->dev;kfree(desc);}list_for_each_entry_safe(desc, temp_desc, &intc_parent_list, list) {list_del(&desc->list);kfree(desc);}
err:list_for_each_entry_safe(desc, temp_desc, &intc_desc_list, list) {list_del(&desc->list);of_node_put(desc->dev);kfree(desc);}
}

time_init

time_init会调用machine_desc->init_time,根据上面DT_MACHINE_START的定义,也即mmp_init_time;主要使用of_clk_init和timer_probe初始化时钟

static void __init mmp_init_time(void)
{enable_pxawdt_clock();/* Reset and Enable the Timer set0, Select the timer set's fast clock source as 3.25MHz *//* The actual clock value for each timer in the timer set are decided by the TMR_CCR *//* The MIPSRAM actually uses the 2nd timer of timer set0 to count LPM periods*/__raw_writel(APBC_APBCLK | APBC_RST, APBC_MMPX_TIMER0);__raw_writel(APBC_APBCLK | APBC_FNCLK | APBC_FNCLKSEL(3),APBC_MMPX_TIMER0);#if 0/* Reset and Enable the Timer set0, Select the timer set's fast clock source as 3.25MHz *//* MIPSRAM uses this timer to time LPM periods */__raw_writel(APBC_APBCLK | APBC_RST, APBC_MMPX_TIMER1);__raw_writel(APBC_APBCLK | APBC_FNCLK | APBC_FNCLKSEL(3),APBC_MMPX_TIMER1);
#else/** the CP timer(used as cp watchdog timer) is not routed to AP, AP uses* cp timer as timer tick, cp uses APB Timer1(d4016000)* cp timer is fixed at 13MHZ, here to enable and reset the core timer*/__raw_writel(APBC_APBCLK | APBC_FNCLK | APBC_RST, APBCP_TICER);__raw_writel(APBC_APBCLK | APBC_FNCLK, APBCP_TICER);
#endifof_clk_init(NULL);timer_probe();coresight_debug_init();
}

of_clk_init

CLK_OF_DECLARE(asr1803_clk, "asr,asr1803-clock", asr1803_clk_init);经过_OF_DECLARE展开,就会产生__clk_of_table;然后通过for_each_matching_node_and_match 来解析设备树节点,并在后续主动调用.data,也即asr1803_clk_init;它由晶振+pll给cpu核心和外设产生时钟;主要代码为:

matches = &__clk_of_table; (匹配)

parent->clk_init_cb = match->data; (初始化回调函数)

clk_provider->clk_init_cb(clk_provider->np); (执行回调)

void __init of_clk_init(const struct of_device_id *matches)
{const struct of_device_id *match;struct device_node *np;struct clock_provider *clk_provider, *next;bool is_init_done;bool force = false;LIST_HEAD(clk_provider_list);if (!matches)matches = &__clk_of_table;/* First prepare the list of the clocks providers */for_each_matching_node_and_match(np, matches, &match) {struct clock_provider *parent;if (!of_device_is_available(np))continue;parent = kzalloc(sizeof(*parent), GFP_KERNEL);if (!parent) {list_for_each_entry_safe(clk_provider, next,&clk_provider_list, node) {list_del(&clk_provider->node);of_node_put(clk_provider->np);kfree(clk_provider);}of_node_put(np);return;}parent->clk_init_cb = match->data;parent->np = of_node_get(np);list_add_tail(&parent->node, &clk_provider_list);}while (!list_empty(&clk_provider_list)) {is_init_done = false;list_for_each_entry_safe(clk_provider, next,&clk_provider_list, node) {if (force || parent_ready(clk_provider->np)) {/* Don't populate platform devices */of_node_set_flag(clk_provider->np,OF_POPULATED);clk_provider->clk_init_cb(clk_provider->np);of_clk_set_defaults(clk_provider->np, true);list_del(&clk_provider->node);of_node_put(clk_provider->np);kfree(clk_provider);is_init_done = true;}}/** We didn't manage to initialize any of the* remaining providers during the last loop, so now we* initialize all the remaining ones unconditionally* in case the clock parent was not mandatory*/if (!is_init_done)force = true;}
}

timer_probe

TIMER_OF_DECLARE(mmp_timer, "mrvl,mmp-timer", mmp_dt_init_timer); 经过_OF_DECLARE展开,就会产生__timer_of_table;然后通过for_each_matching_node_and_match 来解析设备树节点,并在后续主动调用.data,也即mmp_dt_init_timer;主要代码为:

for_each_matching_node_and_match(np, __timer_of_table, &match); (匹配)

init_func_ret = match->data; (初始化回调函数)

init_func_ret(np); (执行回调)

extern struct of_device_id __timer_of_table[];static const struct of_device_id __timer_of_table_sentinel__used __section(__timer_of_table_end);void __init timer_probe(void)
{struct device_node *np;const struct of_device_id *match;of_init_fn_1_ret init_func_ret;unsigned timers = 0;int ret;for_each_matching_node_and_match(np, __timer_of_table, &match) {if (!of_device_is_available(np))continue;init_func_ret = match->data;ret = init_func_ret(np);if (ret) {if (ret != -EPROBE_DEFER)pr_err("Failed to initialize '%pOF': %d\n", np,ret);continue;}timers++;}timers += acpi_probe_device_table(timer);if (!timers)pr_crit("%s: no matching timers found\n", __func__);
}

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

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

相关文章

用友YonSuite打通招银云直联,让企业收付款更便利

在当今数智化浪潮席卷全球的背景下&#xff0c;企业对于高效、便捷的管理系统需求日益增加。作为全球领先的企业云服务与软件提供商&#xff0c;用友始终站在技术前沿&#xff0c;致力于为成长型企业提供全方位的数智化解决方案。 用友网络与招商银行通过联通双方系统&#xf…

YOLOv8改进 | 卷积模块 | 用坐标卷积CoordConv替换Conv

&#x1f4a1;&#x1f4a1;&#x1f4a1;本专栏所有程序均经过测试&#xff0c;可成功执行&#x1f4a1;&#x1f4a1;&#x1f4a1; 专栏目录&#xff1a;《YOLOv8改进有效涨点》专栏介绍 & 专栏目录 | 目前已有40篇内容&#xff0c;内含各种Head检测头、损失函数Loss、B…

消息订阅报错-用户拒绝: errMsg

微信小程序-消息订阅报错 前提问题解决 前提 小程序开发过程中&#xff0c;使用消息订阅功能。 问题 场景一&#xff1a;再注册成功之后进行消息订阅功能。 这里消息订阅功能写在接口请求成功的方法里面。 但是项目运行起来&#xff0c;消息订阅功能并没有被调用起来。 并且…

pytorch保存加载模型的两种方式

这里写目录标题 pytorch保存模型的两种方式开始&#xff1a;定义一个简单的模型方式一&#xff1a;保存和加载模型的状态字典方式二&#xff1a;2保存和加载整个模型 pytorch保存模型的两种方式 开始&#xff1a;定义一个简单的模型 import torch import torch.nn as nn impo…

JSP中实现文件上传下载

JSP中实现文件上传下载 文章目录 JSP中实现文件上传下载前言一、文件上传1、将jar包放入WEB-INF目录下的lib目录2、实现upload.jsp3、实现MyUploadServlet 二、文件下载1.download.jsp2.MyDownloadServlet 前言 在java web开发中&#xff0c;实现单个或多个文件的上传下载&…

期货的心态

1.跌倒之后爬起过程中的收获 失败跌倒了&#xff0c;一定不要自暴自弃&#xff0c;而是要有跌倒不要紧&#xff0c;要相信爬起来的过程中都会有一定的收获&#xff0c;从交易错误中走出并升华&#xff0c;这就会一步步的使我们迈向更高层次的交易。 2.不要让今天的交易影响到明…

智能化立体仓库的种类有哪些?

在仓储运输系统中&#xff0c;自动化立体仓库可充分利用空间储存货物&#xff0c;故而也被称之为高层货架仓库。在实际应用中&#xff0c;自动化仓库系统是不需人工处理的情况下能自动存储和取出物料的系统。那么&#xff0c;智能化立体仓库的种类有哪些&#xff1f;下面就让小…

MySQL相关的19个笔试题

以下是MySQL相关的19个笔试题&#xff0c;涵盖了MySQL的基础知识、SQL语句、索引、事务、优化等方面&#xff1a; 1. MySQL是什么类型的数据库&#xff1f; 答案&#xff1a;MySQL是一个开源的关系型数据库管理系统&#xff08;RDBMS&#xff09;。 2. MySQL支持哪些数据类型…

用Selenium自动化Web应用测试!

在开发和维护Web应用时&#xff0c;测试是确保应用正常运行的关键环节。手动测试不仅费时费力&#xff0c;而且容易出错。而通过使用Selenium&#xff0c;程序员可以轻松模拟用户交互、验证页面元素&#xff0c;从而自动化测试过程&#xff0c;提升测试效率和准确性。 解决的问…

第十五章 观察者模式

目录 1 观察者模式介绍 2 观察者模式原理 3 观察者模式实现 4 观察者模式应用实例 5 观察者模式总结 1 观察者模式介绍 观察者模式的应用场景非常广泛&#xff0c;小到代码层面的解耦&#xff0c;大到架构层面的系统解耦&#xff0c;再或者 一些产品的设计思路&#xff0c…

FREERTOS中,队列按键捕获

队列消息获取的按键任务优先级 < 队列消息释放的按键任务优先级 会出现错误&#xff0c;必须先按KEY1&#xff0c;才能按KEY0 解决方法&#xff1a;修改任务的优先级&#xff0c;队列消息获取的按键任务优先级 > 队列消息释放的按键任务优先级

Javaweb之web开发概述

一、Javaweb简介 用Java技术来解决相关web互联网领域的技术栈.使用JAVAEE技术体系开发企业级互联网项目. 项目规模和架构模式与JAVASE阶段有着很大的差别. 在互联网项目下,首先需要明白客户端和服务器的概念 客户端 :与用户进行交互&#xff0c;用于接收用户的输入(操作)、展示…

Zookeeper ZNode 数据结构原理

ZNode 学习指南 1. ZNode 基本概念 什么是 ZNode:ZNode 是 Zookeeper 中的数据节点。它类似于文件系统中的文件和目录,ZNode 既可以保存数据又可以作为其他 ZNode 的父节点。ZNode 的路径:每个 ZNode 在 Zookeeper 命名空间中都有一个唯一的路径,如 /app1/config。2. ZNod…

❤️‍❤️‍❤️‍FlyFlow 工作流:支持字典管理并支持表单引用

FlyFlow 介绍 官网地址&#xff1a;www.flyflow.cc 演示网址&#xff1a;pro.flyflow.cc FlyFlow 借鉴了钉钉与飞书的界面设计理念&#xff0c;致力于打造一款用户友好、快速上手的工作流程工具。相较于传统的基于 BPMN.js 的工作流引擎&#xff0c;我们提供的解决方案显著简…

20240619每日小程序-------朋友想开发微信小程序,那就搞一把demo

下载开发工具 hbuildX 微信开发者工具 随便搞个开源项目 会员小程序 下载后导入到hbuildX 安装依赖 npm i 安装hbuildX插件 工具—》插件安装 推荐安装&#xff1a; 微信小程序一键打包插件sass编译 启动 选择5.用微信开发者工具启动 报错不要怕 比如&#xff1a…

微信小程序生命周期分为3种:页面级别,应用级别,组件级别

应用级别&#xff1a; onLaunch&#xff0c;onShow&#xff0c;onHide 页面级别&#xff1a;onLoad&#xff0c;onShow&#xff0c;onReady&#xff0c;onHide&#xff0c;onUnload 组件级别&#xff1a;

硕思闪客精灵_2024最新版下载-闪客精灵软件下载_闪客精灵应用软件

​不同领域的应用证明了能够解析Flash动画片中的视频文件并以*.fla格式进行导出。人所共知的是支持预览和播放所选的Flash动画片或元素。我们都知道除了将静态文字恢复为文本外&#xff0c;硕思闪客精灵提供了将它转换为矢量图的功能。相信大家都认同闪客精灵专业版的优势&…

【鲸吞法实操记录】开始挑课题

研一下的文档 小论文 主题方向未定 根据这两篇博文&#xff0c;总结出这个领域的一些key words (1条消息) 3D目标检测论文汇总_light169的博客-CSDN博客 (1条消息) 基于深度学习目标姿态估计的论文一览_light169的博客-CSDN博客_姿态估计论文 以下key words 应该是涵盖了 …

通用VS垂直,落地场景的抉择。

随着人工智能技术的飞速发展&#xff0c;大模型已成为推动产业智能化升级的核心力量。在这个竞争激烈的战场中&#xff0c;通用大模型和垂直大模型各自展现出独特的优势&#xff0c;引发了业界的广泛关注。那么&#xff0c;对于大模型的落地应用&#xff0c;谁将率先形成绝对优…

使用 ngrok 实现内网穿透

引言 ngrok 是一款流行的内网穿透工具&#xff0c;它允许你将本地服务器暴露到公网上。这在开发过程中非常有用&#xff0c;特别是当你需要从不同的地方访问你的应用或者服务时。本文将指导你如何安装和使用 ngrok&#xff0c;以及如何通过它实现内网穿透。 前提条件 Window…