进程切换的本质是什么?

1a432518b0bbaec75dad670ab1d28af5.gif

作者 | 陆小凤

来源 | 码农的荒岛求生

我们都知道操作系统最重要的功能之一是多任务能力,也就是可以运行超过CPU数量的程序——即进程,要想实现这一功能就必须具备将有限的CPU资源在多个进程之间分配的能力,在程序员看来,我们的程序在一直运行,而在CPU看来程序其实在“走走停停”,程序的一走一停就涉及到进程切换,那么进程切换的本质是什么呢?

从本质上讲,函数调用和进程切换是非常类似的,有的同学可能会有疑问,这怎么可能呢?别着急,看完这篇你就明白啦。

函数调用

我们先来看一下函数调用,函数调用是这样的,A函数调用B函数,当B函数执行完成时会跳转回A函数(此时A函数和B函数位于同一个进程):

void B() {...
}
void A() {...
}

这个过程是这样的:

f5bd19aca5f508507d364effc3d3b95a.png

B函数执行完成后会将控制权转给A,所谓控制权是指告诉CPU继续执行函数A。

but,你有没有想过,A函数调用B函数,B函数执行完成时一定要跳转回A函数吗?不一定的,既然B函数可以将控制权转给A那么就能将控制权转给函数C。

听上去很神奇有没有,A函数调用B函数,当B函数执行完成时竟然可以跳转到C函数,可这该怎样做到呢?让我们来看一段神奇的代码。

一段神奇的代码

有这样一段代码:

#include <stdio.h>
#include <stdlib.h>void funcC() {printf("jump to funcC !!!\n") ;exit(-1) ;
}void funcB() {long *p = NULL ;p = (long*)&p ;*(p+2) = (long)funcC ;
}void funcA() {funcB();
}int main() {funcA() ;return 0 ;
}

想一想这段代码运行后会输出什么?

有的同学可能会说main函数调用了funcA,funcA函数调用了funcB,funcB函数看上就是一堆赋值,执行完成后返回了funcA,funcA又返回main函数,因此执行完毕后什么都不会输出。

真的是这样的吗?让我们编译运行一下(小风哥使用的是5.2.0版gcc,64位机器,未开启编译优化,不同编译器版本运行效果可能不同)。

$ ./a.out
jump to funcC !!!

有的同学也许会大吃一惊,这怎么可能!!!

这段明明没有调用funcC,可为什么funcC函数却运行了?

程序员经常说“代码之中没有秘密”,这句话不全对,应该是“机器指令中没有秘密”,后来我想了想,这句话也不全对,因为对我们来说CPU是如何执行机器指令这回事其实对我们来说是黑盒的,我们只能从大体的原理来说CPU是怎样执行一条机器指令的,但这里真正的细节只有处理器生产商比如intel/AMD等知道,而一些魔鬼恰恰就在这些细节中。

魔鬼在细节

扯远了,让我们回到这篇文章的主题,先来看看生成的机器指令是什么样的:

0000000000400586 <funcC>:400586:       55                      push   %rbp400587:       48 89 e5                mov    %rsp,%rbp40058a:       bf 74 06 40 00          mov    $0x400674,%edi40058f:       e8 bc fe ff ff          callq  400450 <puts@plt>400594:       bf ff ff ff ff          mov    $0xffffffff,%edi400599:       e8 e2 fe ff ff          callq  400480 <exit@plt>000000000040059e <funcB>:40059e:       55                      push   %rbp40059f:       48 89 e5                mov    %rsp,%rbp4005a2:       48 c7 45 f8 00 00 00    movq   $0x0,-0x8(%rbp)4005a9:       00 4005aa:       48 8d 45 f8             lea    -0x8(%rbp),%rax4005ae:       48 89 45 f8             mov    %rax,-0x8(%rbp)4005b2:       48 8b 45 f8             mov    -0x8(%rbp),%rax4005b6:       48 83 c0 10             add    $0x10,%rax4005ba:       ba 86 05 40 00          mov    $0x400586,%edx4005bf:       48 89 10                mov    %rdx,(%rax)4005c2:       90                      nop4005c3:       5d                      pop    %rbp4005c4:       c3                      retq

这些指令在说什么呢?

我们先来看普通的函数调用:

851531f242d93cffcf6a619336ddbb51.png

当函数B执行完毕后此时的栈帧为:

08ff3afdc4d89c9fff0f31029774fbe3.png

函数B的最后一条机器指令通常为:ret,这条指令的目的是将当前栈顶的内容弹出到%rip寄存器中,CPU会根据rip中的值从内存中取出指令并执行,显然ret指令会将之前保存的返回地址放入rip寄存器中,这样CPU就可以继续执行A函数中的后续代码了,也就是++a这行代码。

有的同学可能已经看出来,如果我们有办法修改A栈帧上的返回地址不就能实现“指哪打哪”了吗?

实际上代码中“*(p+2) = (long)funcC ;”这行会将本来指向funcB的返回地址修改为指向funcC,即:

46d14dc2144ed9c09aa3f8e74e25d39d.png

这样当funcB函数运行完成后会直接跳转到funcC函数,从而实现可控的执行流切换,进程切换的本质与此别无二致,只不过进程切换需要连带着把栈也切换过去(以及地址空间),同时还会保存被切换进程的上下文。

有的同学可能已经看出来了,上述过程叫做缓冲区溢出攻击,要实现的目的和进程切换一样:实现控制权的转移,只不过缓冲区溢出攻击是非法的,不符合预期的(符合黑客的预期,但不符合操作系统设计者制定的游戏规则),而进程切换是合法的,符合预期的(符合操作系统设计者的预期)。

而有时,(真正意义上的)黑客与操作系统设计者其实是一伙人55923757d4331c29b254f0474b93b30b.png

怎么样,现在你应该对进程切换有较为直观的认知了吧,当然真实的进程切换绝不像这里讲解的这样简单呦~

5c2648af9d019883ff64fbb63a5719ff.gif

往期推荐

高并发下的 HashMap 为什么会死循环

操作系统如何实现:什么是宏内核、微内核

Redis 内存满了怎么办?这样置才正确!

如何在 Kubernetes Pod 内进行网络抓包

ed532726ed2c7bb5c58c5166fd26bb3b.gif

点分享

29b39b6fdba87d0bbed35517a2952534.gif

点收藏

a3cf127267e9723391db4c39f66f684b.gif

点点赞

4c748f8faaf4bdafa82c2ba44de4c238.gif

点在看

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

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

相关文章

lol1.7更新服务器维护,lol今天停机维护到几点11日7.1版本停机更新公告

lol今天停机维护到几点&#xff0c;lol1月11日停机维护更新公告&#xff0c;lol今天怎么进不去2017?下面小编将英雄联盟发布的停机公告详细给大家介绍。lol今天停机维护到几点1月11日早7点30分全区停机维护&#xff0c;预计停机时间为07:30-12&#xff1a;0011日7.1版本停机更…

Log4j漏洞不仅仅是修复,更需要构建有效预警机制

简介&#xff1a;软件的漏洞有时不可避免&#xff0c;根据Gartner的相关统计&#xff0c;到 2025 年&#xff0c;30% 的关键信息基础设施组织将遇到安全漏洞。日志服务SLS&#xff0c;可帮助快速部署一个预警机制&#xff0c;使得漏洞被利用时可以快速发现并及时响应。通过使用…

太强了!这款开源终端工具可查询 IP 信息~

作者 | JackTian来源 | 杰哥的IT之旅在 Linux 下&#xff0c;有dig、nslookup、traceroute等多种非常实用的网络调试工具。dig&#xff1a;是常用的域名查询工具&#xff0c;可以用来测试域名是否正常。nslookup&#xff1a;是常用的域名查询工具&#xff0c;也就是查 DNS 信息…

顺序写磁盘比随机写内存_深入理解 linux磁盘顺序写、随机写

一、前言随机写会导致磁头不停地换道&#xff0c;造成效率的极大降低&#xff1b;顺序写磁头几乎不用换道&#xff0c;或者换道的时间很短。本文来讨论一下两者具体的差别以及相应的内核调用。二、环境准备三、fio介绍通过fio测试&#xff0c;能够反映在读写中的状态&#xff0…

为余势负天工背,云原生内存数据库Tair助力用户体验优化

简介&#xff1a;作为双11大促承载流量洪峰的利器&#xff0c;Tair支撑了电商交易核心体验场景。不仅在数十亿QPS的峰值下保持着亚毫秒级别的顺滑延迟&#xff0c;同时在电商交易核心体验场景上也做出了技术创新。 作者 | 漠冰 来源 | 阿里技术公众号 作为双11大促承载流量洪峰…

【视频特辑】数据分析师必备,快速制作一张强大好用的大宽表

简介&#xff1a;随着企业数字化进程的逐步推进&#xff0c;在日常经营过程当中会沉淀下越来越多的数据信息。 每当想做数据分析的时候&#xff0c;就会发现想要的指标分散在不同的数据源、数据集、数据表当中。 Quick BI的数据关联功能&#xff0c;可以帮助数据分析师快速将指…

读取硬盘前的准备工作有哪些?

作者 | 闪客sun来源 | 低并发编程读取硬盘数据到内存中&#xff0c;是操作系统的一个基础功能。读取硬盘需要有块设备驱动程序&#xff0c;而以文件的方式来读取则还有要再上面包一层文件系统。把读出来的数据放到内存&#xff0c;就涉及到内存中缓冲区的管理。上面说的每一件事…

【视频特辑】提效神器,如何用Quick BI高效配置员工的用数权限

简介&#xff1a;随着企业数字化进程逐步加速&#xff0c;企业所产生和积累的数据资源日益增多。每当员工的用数权限发生变动&#xff0c;管理员都需要进行复杂繁琐的重复性配置流程&#xff0c;不仅耗时耗力还容易出错。 如何能便捷地对员工用数权限进行高效管理&#xff1f;试…

让容器跑得更快:CPU Burst 技术实践

简介&#xff1a;让人讨厌的 CPU 限流影响容器运行&#xff0c;有时人们不得不牺牲容器部署密度来避免 CPU 限流出现。我们设计的 CPU Burst 技术既能保证容器运行服务质量&#xff0c;又不降低容器部署密度。CPU Burst 特性已合入 Linux 5.14&#xff0c;Anolis OS 8.2、Aliba…

实时数仓Hologres首次走进阿里淘特双11

简介&#xff1a;这是淘特在阿里巴巴参与的第二个双11大促&#xff0c;大促期间累计超过上千万消费者在此买到心仪的商品&#xff0c;数百万家商家因为淘特而变得不同&#xff0c;未来&#xff0c;淘特也将会继续更好的服务于下沉市场&#xff0c;让惠民走近千万家。 2021年11…

Cluster 集群能支撑的数据有多大?

作者 | 码哥字节来源 | 码哥字节本文将对集群的节点、槽指派、命令执行、重新分片、转向、故障转移、消息等各个方面进行深入拆解。目的在于掌握什么是 Cluster &#xff1f;Cluster 分片原理&#xff0c;客户端定位数据原理、故障切换&#xff0c;选主&#xff0c;什么场景使用…

All in one:如何搭建端到端可观测体系

简介&#xff1a;一文看懂可观测&#xff01; 作者&#xff1a;西杰 & 白玙 可观测的前生今世 系统的可观测与故障可分析作为系统运维中重要的衡量标准&#xff0c;随着系统在架构、资源单位、资源获取方式、通信方式演进过程&#xff0c;遇到了巨大挑战。而这些挑战&am…

链路分析 K.O “五大经典问题”

简介&#xff1a;链路分析是基于已存储的全量链路明细数据&#xff0c;自由组合筛选条件与聚合维度进行实时分析&#xff0c;可以满足不同场景的自定义诊断需求。 作者&#xff1a;涯海 链路追踪的 “第三种玩法” 提起链路追踪&#xff0c;大家会很自然的想到使用调用链排查…

Kubernetes 上容器的启动顺序如何把控?

作者 | AddoZhang来源 | 云原生指北为什么要做容器启动顺序控制&#xff1f;我们都知道 Pod 中除了 init-container 之外&#xff0c;是允许添加多个容器的。类似 TektonCD 中 task 和 step 的概念就分别与 pod 和 container 对应&#xff0c;而 step 是按照顺序执行的。此外还…

一文说清linux system load

简介&#xff1a;双十一压测过程中&#xff0c;常见的问题之一就是load 飙高&#xff0c;通常这个时候业务上都有受影响&#xff0c;比如服务rt飙高&#xff0c;比如机器无法登录&#xff0c;比如机器上执行命令hang住等等。本文就来说说&#xff0c;什么是load&#xff0c;loa…

KubeDL 0.4.0 - Kubernetes AI 模型版本管理与追踪

简介&#xff1a;欢迎更多的用户试用 KubeDL&#xff0c;并向我们提出宝贵的意见&#xff0c;也期待有更多的开发者关注以及参与 KubeDL 社区的建设&#xff01; 作者&#xff1a;陈裘凯&#xff08; 求索&#xff09; 前言 KubeDL 是阿里开源的基于 Kubernetes 的 AI 工作负…

上云一时爽,遇坑泪两行

如今&#xff0c;企业的数字化转型进程已经进入了“快车道”&#xff0c;各行各业基于自身业务发展与变革的需要&#xff0c;为整体数字化转型带来了更多要求。企业纷纷依托云原生、低代码、大数据、人工智能等技术手段积极加入这场没有硝烟的战争。 对于传统企业而言&#xf…

读研期间一定得看论文吗_读研期间各阶段的目标和任务,你明确吗?

读研期间一般都要经历上课、论文材料的收集、论文的开题、发表小论文、毕业论文的答辩、找工作或考博士等几个关键环节。在校期间&#xff0c;我们不仅要完成以上的全部工作&#xff0c;还需要不断地学习正确的价值观和人生观&#xff0c;学会科学的思考。如何让自己的研究生生…

Spring Boot Serverless 实战系列“架构篇” | 光速入门函数计算

简介&#xff1a;如何以 Serverless 的方式运行 Spring Boot 应用&#xff1f; 作者&#xff1a;西流&#xff08;阿里云函数计算专家&#xff09; Spring Boot 是基于 Java Spring 框架的套件&#xff0c;它预装了 Spring 一系列的组件&#xff0c;开发者只需要很少的配置即可…

从 “香农熵” 到 “告警降噪” ,如何提升告警精度?

简介&#xff1a;ARMS 智能降噪功能依托于 NLP 算法和信息熵理论建立模型&#xff0c;从大量历史告警事件中去挖掘这些事件的模式规律。当实时事件触发后&#xff0c;实时为每一条事件打上信息熵值与噪音识别的标签&#xff0c;帮助用户快速识别事件重要性。 作者&#xff1a;…