操作系统进程学习(Linux 内核学习笔记)

操作系统进程学习(Linux 内核学习笔记)

进程优先级

并非所有进程都具有相同的重要性。除了大多数我们所熟悉的进程优先级之外,进程还有不同的关键度类别,以满足不同需求。首先进程比较粗糙的划分,进程可以分为实时进程非实时进程(普通进程)

实时进程优先级(0-99)都比普通 进程的优先级(100-139)高。当系统中有实时进程运行时,普通进程几乎无法分到时间片(只能分到5%的CPU时间)。

通过时间片运行进程

进程系统调用

​ 讨论fork和exec系列系统调用的实现。通常这些调用不是由应用程序直接发出的,而是通过一个中间层调用,即负责与内核通信的C标准库。从用户状态切换到核心态的方法,依不同的体系结构而各有不同

用户发起一个新进程之后,CPU为进程分配资源,并将硬盘数据读到内存中去,但是用户进程是一个应用级程序,无法直接与CPU进行交互,所以通过系统调用(system_call),加载完成内核就会退出CPU,用户进程执行完毕后,内核进入CPU将用户进程移除,下一个同上image-20210903163111488

  1. 进程复制

    (1) fork是重量级调用,因为它建立了父进程的一个完整副本,然后作为子进程执行。 为减少与该调用相关的工作量,Linux使用了写时复制(copy-on-write)1技术。

    (2) vfork类似于fork,但并不创建父进程数据的副本。相反,父子进程之间共享数据。 这节省了大量CPU时间(如果一个进程操纵共享数据,则另一个会自动注意到)。

    (3) clone产生线程,可以对父子进程之间的共享、复制进行精确控制。

调度器分析

  1. 调度器及其功能

    内核中用来安排进程执行的模块称为调度器(scheduler),它可以切换进程状态 (process state)。例如执行、可中断睡眠、不可中断睡眠、退出、暂停等。 调度器是CPU中央处理器的管理员,主要负责完成做两件事情:一、选择某些就绪进 程来执行,二是打断某些执行的进程让它们变为就绪状态。调度器分配CPU时间的基本依据就 是进程的优先级。上下文 切换(context switch):将进程在CPU中切换执行的过程,内核承担 此任务,负责重建和存储被切换掉之前的CPU状态

主要有两个函数可以获得线程设置的最高和最低优先级:

int sched_get_priority_max(int); // 获取实时优先级最大值
int sched_get_priority_min(int); // 获取实时优先级最小值

设置与获取优先级各个主要函数:

int pthread_attr_setschedparam(pthread_attr_t*,const struct sched_param);//创建线程优先级
int pthread_attr_getschedparam(pthread_attr_t*,const struct sched_param);//获取线程优先级

实战:

#include <assert.h>
#include <iostream>
#include <pthread.h>
#include <sched.h>
#include <stdio.h>static int get_thread_policy(pthread_attr_t *attr) //获取线程调度策略
{int plicy;int rs = pthread_attr_getschedpolicy(attr, &plicy);assert(rs == 0);switch (plicy){case SCHED_FIFO:printf("policy = FIFO");break;case SCHED_RR:printf("policy = RR");break;case SCHED_OTHER:printf("policy=OTHER");break;default:printf("policy = unknown");break;}return plicy;
}static void show_thread_priority(pthread_attr_t *attr, int policy)
{int priority = sched_get_priority_max(policy);assert(priority != -1);printf("max_priority=%d\n", priority);priority = sched_get_priority_min(policy);assert(priority != -1);printf("min_priority=%d\n", priority);
}static int get_thread_priority(pthread_attr_t *attr) //获取线程优先级
{sched_param param;int rs = pthread_attr_getschedparam(attr, &param);assert(rs == 0);printf("priority = %d", param.sched_priority);return param.sched_priority;
}
static void set_thread_policy(pthread_attr_t *attr, int policy)
{int rs = pthread_attr_setschedpolicy(attr, policy);assert(rs == 0);get_thread_policy(attr);
}int main(int argc, char const *argv[])
{pthread_attr_t attr;struct sched_param sched;int rs = pthread_attr_init(&attr);assert(rs == 0);int plicy = get_thread_policy(&attr);printf("输出进程优先级\n");show_thread_priority(&attr, plicy);printf("输出FIFO优先级");show_thread_priority(&attr, SCHED_FIFO);printf("输出RR优先级\n");show_thread_priority(&attr, SCHED_RR);printf("输出当前线程优先级");int priority = get_thread_priority(&attr);printf("设置线程策略\n");printf("设置FIFO策略");set_thread_policy(&attr, SCHED_FIFO);printf("设置RR策略");set_thread_policy(&attr, SCHED_RR);printf("还原线程属性\n");set_thread_policy(&attr, plicy);rs = pthread_attr_destroy(&attr);assert(rs == 0);return 0;
}

实战2

#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>void threadfunc1()
{sleep(1);int policy;struct sched_param praram;pthread_getschedparam(pthread_self(), &policy, &praram);if (policy == SCHED_OTHER)printf("SCHED_OTHER.\n");if (policy == SCHED_RR);printf("SCHED_RR 1.\n");if (policy == SCHED_FIFO)printf("SCHED_FIFO.\n");for (int i = 1; i <= 10; i++){for (int j = 1; j < 4000000; j++){}printf("Threadfunc1.\n");}printf("pthreadfunc1 EXIT.\n");
}void threadfunc2()
{sleep(1);int policy;struct sched_param praram;pthread_getschedparam(pthread_self(), &policy, &praram);if (policy == SCHED_OTHER)printf("SCHED_OTHER.\n");if (policy == SCHED_RR);printf("SCHED_RR 1.\n");if (policy == SCHED_FIFO)printf("SCHED_FIFO.\n");for (int i = 1; i <= 10; i++){for (int j = 1; j < 4000000; j++){}printf("Threadfunc2.\n");}printf("pthreadfunc2 EXIT.\n");
}void threadfunc3()
{sleep(1);int policy;struct sched_param praram;pthread_getschedparam(pthread_self(), &policy, &praram);if (policy == SCHED_OTHER)printf("SCHED_OTHER.\n");if (policy == SCHED_RR);printf("SCHED_RR 1.\n");if (policy == SCHED_FIFO)printf("SCHED_FIFO.\n");for (int i = 1; i <= 10; i++){for (int j = 1; j < 4000000; j++){}printf("Threadfunc3.\n");}printf("pthreadfunc3 EXIT.\n");
}int main()
{int i;i = getuid();if (i == 0)printf("the current user is root.\n");elseprintf("the current user is not root.\n");pthread_t ppid1, ppid2, ppid3;struct sched_param param;pthread_attr_t attr1, attr2, attr3;pthread_attr_init(&attr2);pthread_attr_init(&attr1);pthread_attr_init(&attr3);param.sched_priority = 51;pthread_attr_setschedpolicy(&attr3, SCHED_RR);pthread_attr_setschedparam(&attr3, &param);pthread_attr_setinheritsched(&attr3, PTHREAD_EXPLICIT_SCHED);param.sched_priority = 22;pthread_attr_setschedpolicy(&attr2, SCHED_RR);pthread_attr_setschedparam(&attr2, &param);pthread_attr_setinheritsched(&attr2, PTHREAD_EXPLICIT_SCHED);pthread_create(&ppid3, &attr1, (void *)threadfunc3, NULL);pthread_create(&ppid2, &attr2, (void *)threadfunc2, NULL);pthread_create(&ppid1, &attr3, (void *)threadfunc1, NULL);pthread_join(ppid3, NULL);pthread_join(ppid2, NULL);pthread_join(ppid1, NULL);pthread_attr_destroy(&attr3);pthread_attr_destroy(&attr2);pthread_attr_destroy(&attr1);return 0;
}

RCU机制

​ RCU英文全称为Read-Copy-Update,顾名思义就是 “读 - 拷贝-更新”,是内核中重要 的同步机制。Linux内核已有原子操作、读写信号量等等锁机制,为何会单独设计一个比较复杂的新机制?

RCU原理

​ RCU记录所有指向共享数据的指针的使用者,当要修改该共享数据时,首先创建一个 副本,在副本中修改。所有读访问线程都离开读临界区之后 ,指针指向新的修改后副本的指针,并且删除旧数据

  1. 写者修改对象的过程:首先生成一个副本,让后更新副本,最后使用新的对象替换旧的对象,在写者执行复制更新的时候读者可以读数据
  2. 写者删除对象:必须等到所有访问的读者访问结束,才能执行销毁操作,RCU技术的关键:如何判断所有的读者已经结束访问,等待所有读者访问时间结束的时间称为宽限期
  3. RCU读者并不需要与写进行同步,读者和写者能够并发执行,RCU目标最大程序来减少读者的开销,因为也经常使用于读者性能要求高的场合

RCU优点:读者开销少,不需要锁,原子指令,内存屏障等,无死锁问题,无内存泄露风险

RCU缺点:写者同步开销比较大,写着之间需要互斥处理,同步机制复杂

image-20210905142136565

内核内存布局

​ ARM64架构处理器采用48位物理寻址机制,最大可寻找256TB的物理地址空间。对于 目前应用完全足够,不需要扩展到64位的物理寻址。虚拟地址也同样最大支持48位寻址,所以 在处理器架构设计上,把虚拟地址空间划分为两个空间,每个空间最大支持256TB,linux内核 在大多数体系结构上都把两个地址划分为:用户空间和内核空间。

用户空间:0x0000_0000_0000_0000至0x0000_ffff_ffff_ffff。

内核空间:0xffff_0000_0000_0000至0xffff_ffff_ffff_ffff

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

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

相关文章

gcc对C语言的扩展:语句内嵌表达式(statement-embedded expression)

在gnu c 中&#xff0c;用括号将复合语句括起来也形成了表达式。他允许你在一个表达式内使用循环&#xff0c;跳转和局部变量。一个复合语句是用大括号{}括起来的一组语句。在包含语句的表达式这种结构中&#xff0c;再用括号( )将大括号括起来,例如&#xff1a;({ int y foo …

react学习(56)--常见HTTP错误

200: 服务器成功返回请求的数据。,201: 新建或修改数据成功。,202: 一个请求已经进入后台排队&#xff08;异步任务&#xff09;。,204: 删除数据成功。,400: 发出的请求有错误&#xff0c;服务器没有进行新建或修改数据的操作。,401: 用户没有权限&#xff08;令牌、用户名、密…

C#二叉树递归实现

二叉树类(binaryTree.cs) using System; namespace binary_tree_demo { class BinaryTreeNode where T : IComparable { public BinaryTreeNode() { left null; right null; } public BinaryTreeNode(BinaryTreeNode l, BinaryTreeNode r) { left l; right r; } public Bin…

Linux守护进程的创建(结合nginx框架)

Linux守护进程的创建(结合nginx框架) 先介绍几个相关函数: int dup2(arg1,arg2):参数一指向的内容赋给参数二,shi的参数二也能访问参数一所指向的内容,并返回新的描述符 int fork()创建子进程,返回值-1:创建失败 返回值0:子进程 返回其他:父进程 setsid()调用成功后&#x…

用aspnet_compiler发布网站 (转载:My way of my life )

在asp.net 2.0模型中&#xff0c;vs2005已经完全脱离了编译而成为了一个彻底的ide.算是一个不小的改动。其中更是取消了有关Web Application的概念&#xff0c;使得习惯了vs2003的人刚开始的时候会有一些摸不着头脑。下面简单说一下我在使用过程中自己总结的&#xff0c;算是一…

react学习(57)--map赋值

<Radio.Group>{linksList?.map((item) > (<Radio key{item.key} value{item.key}>{item.value}</Radio>))}</Radio.Group>

使用 dojo/query

在本篇文章中&#xff0c;我们将了解DOM的查询以及如何运用dojo/query这个模块来轻松地选择节点并操作他们。 入门指南 在操作DOM的过程中&#xff0c;如何快速高效地检索出DOM节点显得相当重要。我们在Dojo DOM Functions中已经熟悉了 dom.byId&#xff0c;然而&#xff0c;在…

【Linux内核】虚拟地址空间布局架构

虚拟地址空间布局架构(Linux内核学习) 1.Linux内核整体架构及子系统 内核对下管理硬件,对上通过运行时库对应用提供服务 用户空间 使用malloc()分配内存通过free()释放内存 内核空间 虚拟进程负责从进程的虚拟地址空间分配虚拟页,sys_brk来扩大或收缩堆,sys_mmap负责在内存映…

天凉了,大家多穿衣服

这两天天气转凉&#xff0c;我还穿夏天的衬衫&#xff0c;结果今晚回来发现喉咙不舒服&#xff0c;只好去买药了。大家要保重身体呀&#xff01;

[开源]C#中开源软件大汇总(外国的)

一、博客类项目 1.SubText 项目介绍&#xff1a;Subtext 是一个个人博客发布平台&#xff0c;详细的介绍请进SubText 项目分类&#xff1a;博客 项目license:BSD License 项目主页&#xff1a;http://subtextproject.com/ 2.BlogEngine.net 项目介绍&#xff1a;详细的介绍请进…

【Linux内核】内存映射原理

【Linux内核】内存映射原理 物理地址空间 物理地址是处理器在总线上能看到的地址,使用RISC(Reduced Instruction Set Computing精简指令集)的处理器通常只实现一个物理地址空间,外围设备和物理内存使用统一的物理空间, 有些架构的处理器把分配给外围设备的物理地址称为设备内存…

react学习(59)--this.props语法糖

{...this.props}是props所提供的语法糖&#xff0c;可以将父组件的所有属性复制给子组件

我的名字

李桃春风有名流龙翔长空耀九州昌隆盛世舞文墨无拘无束无忧&#xff08;小弟自吹自擂&#xff0c;虚荣一番&#xff09;转载于:https://blog.51cto.com/73945/7773

div 居中嵌套

将一个 DIV 嵌套进另一个 DIV 容器&#xff0c;并保持水平居中、垂直居中&#xff0c;可使用以下代码&#xff1a; <html> <head><title>div居中嵌套</title><style type"text/css">.big{width: 800px;height: 500px;background: #333…

javascript学习系列(1):数组中的map方法

最好的种树是十年前,其次是现在。歌谣 每天一个前端小知识 提醒你改好好学习了 知乎博主 csdn博主 b站博主 放弃很容易但是坚持一定很酷 我是歌谣 喜欢就一键三连咯 你得点赞是对歌谣最大的鼓励 1前言 在我们的日常开发中 不免会有很多需要处理数据的方法 本节主要说一说m…

spoolsv.exe占cpu 99%的解决方法(转)

前几天&#xff0c;太太的笔记本遇到了spoolsv.exe占cpu 99%的的问题&#xff0c;要我帮她重新安装操作系统。我是懒得重装的&#xff0c;所以在网上搜索了一下&#xff1a;大部分关于这个问题的文章都是将相关打印后台服务给禁用解决的&#xff0c;可是这样就不能打印了&#…

【Linux内核】物理内存组织结构

【Linux内核】物理内存组织结构 系统调用mmap 物理内存组织结构 体系结构 目前多处理器系统有两种体系结构&#xff1a; 1&#xff09;非一致内存访问&#xff08;Non-Unit Memory Access&#xff0c;NUMA&#xff09;&#xff1a;指内存被划分成多个 内存节点的多处理器系…

thinkphp __PUBLIC__的定义 __ROOT__等常量的定义

__TMPL__ > APP_TMPL_PATH, // 项目模板目录__ROOT__ > __ROOT__, // 当前网站地址__APP__ > __APP__, // 当前项目地址__GROUP__ > defined(GROUP_NAME)?__GROUP__:__APP__,__ACTION__ > __ACTION__, // 当前操…

面试准备勿重“难”轻“易”

眼下&#xff0c;有些大中专毕业生在做应聘面试准备时&#xff0c;往往把事情想得太复杂&#xff0c;把用人单位考官提出的问题想得过于难&#xff0c;于是在做面试准备时重“难”轻“易”&#xff0c;把精力都放在了高难度问题上&#xff0c;而忽视了基础性的理论和技术知识。…