《Operating Systems:Three easy pieces》: 2.2 内存虚拟化

2.2 内存虚拟化

       现在让我们思考内存。现代机器所呈现的物理存储模型非常简单。内存只是一个字节数组;要读取内存,必须指定一个地址才能访问存储在那里的数据;要写入(或更新)内存,还必须指定要写入到给定地址的数据。

/**********************************************************************************
	> File Name: mem.c
	> Author: wq
	> Mail: 1509802731@qq.com 
	> Created Time: 2018年12月26日 星期三 15时18分06秒
	>Function:间隔一秒打印一次P的值,尝试同时运行多个此程序,体会共享内存(对内存的虚拟化)
 **********************************************************************************/

#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
#include<string.h>
int main(int argc,char *argv[])
{
	int *p = malloc(sizeof(int));
	assert(p != NULL);
	 /*%08X表示八位对齐的十六进制格式,如0X00000001*/
	printf("(%d) memory address of p: %08X\n",getpid(), (unsigned) p);
	*p = 0;
	while (1)
	{
		sleep(1);
		*p = *p + 1;
		printf("(%d) p: %d\n",getpid(), *p);
	}
	return 0;
}

       当程序运行时,内存一直在被访问。程序将所有数据结构保存在内存中,并在执行工作的时候,通过各种指令访问它们,例如加载和存储或其他访问内存的显式指令。不要忘记每条程序指令也在内存中;在每次取指令时都要访问内存。让我们看看上面那个程序mem.c,通过调用malloc()函数申请内存。程序输出如下:

图2.6

这个程序做了这样几件事情。首先,通过malloc()函数申请了部分内存。然后打印出了内存地址,再将数字0赋值给最新申请的内存地址中。最后,每延迟一秒,循环递增存储在该地址上的值并打印出来,随之打印的还有该程序的进程号,该进程号是唯一的。

同样,前面的结果不足为奇。新申请的内存地址是00200000(这是原文的地址,我此处分配的地址是09327008,这是随机分配的).随着程序的运行,它缓慢的更新值并输出结果。

图2.7同时运行多个程序


现在,我们再次同时运行多个mem程序来看看会发生什么(图2.7)。翻译这段我发现和作者所说及运行结果不一致,我的运行结果如上图2.7所示,作者原文如下:

想知道为什么吗?请见文末!


       现在,我们再次同时运行多个mem程序来看看会发生什么(译者实际运行情况见图2.7,原文见图2.8)。我们从图2.8中看到,每个运行的程序都在同一个地址(00200000)分配内存,而每个程序似乎都在独立地更新00200000的值!就好像每个运行中的程序都有自己的私有内存,而不是与其他运行中的程序共享相同的物理内存

        实际上,这里所发生的事情,是因为操作系统正在虚拟化内存。 每个进程都访问自己的专用虚拟地址空间(有时只是称为地址空间),操作系统以某种方式映射到机器的物理内存。 内存中的内存引用一个正在运行的程序不会影响其他进程的地址空间(或操作系统本身); 就运行程序而言,它认为享有的是他自己的物理内存。 然而,真相是物理存储是由操作系统管理的共享资源。 这些究竟是如何实现的也在本书第一部分的主题:虚拟化讨论范围之内。

答案揭晓:要成为作者那样子,需要让系统不要随机分配地址。实际上,随机化,可以很好的防御某些安全缺陷。作者在GitHub给出解决方案如下:

Details

One issue with mem.c is that address space randomization is usually on by default. To turn it off:

macOS

From stackoverflow

Just compile/link as follows: gcc -o mem mem.c -Wall -Wl,-no_pie

Linux

From Giovanni Lagorio:

Under Linux you can disable ASLR, without using a debugger, in (at least) two ways:

  • Use the command setarch to run a process with ASLR disabled; I typically run bash, with which I can execute examples, like this: setarch $(uname --machine) --addr-no-randomize /bin/bash
  • Writing 0 into /proc/sys/kernel/randomize_va_space; you need to be root to do this and this change has (a non-permament) effect on the whole system, which is something you probably don't want. I use this one only inside VMs.

还有一位同学将代码修改后如下:

#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
#include<string.h>
#include <unistd.h> 
int main(int argc,char *argv[])
{
	int *p = malloc(sizeof(int));
	pid_t fpid;
	
	assert(p != NULL);
	 /*%08X表示八位对齐的十六进制格式,如0X00000001*/
	printf("(%d) memory address of p: %08X\n",getpid(), (unsigned) p);
	*p = 0;
	fpid=fork();
	if (fpid < 0)  
        printf("error in fork!");  
    else if (fpid == 0) { 
		printf("child (%d) memory address of p: %08X\n",getpid(), (unsigned) p);
		while (1)
		{
			sleep(1);
			*p = *p + 1;
			printf("(%d) p: %d\n",getpid(), *p);
		}

    } 
    else {  
		printf("parent (%d) memory address of p: %08X\n",getpid(), (unsigned) p);
		while (1)
		{
			sleep(1);
			*p = *p + 1;
			printf("(%d) p: %d\n",getpid(), *p);
		}
    } 	
	return 0;
}

运行结果如下:

子进程与父进程共享内存

本节完!

### 回答1: "Operating Systems: Three Easy Pieces"(操作系统:三步轻松学)是一本由Remzi H. Arpaci-Dusseau和Andrea C. Arpaci-Dusseau编写的计算机科学教材。本书旨在介绍操作系统背后的核心原理和概念,以及其在计算机系统中的作用。 该书的内容涵盖了操作系统的多个方面,包括进程管理、内存管理、文件系统、输入输出、虚拟化、并发与并行等。它提供了广泛的范围,旨在帮助读者全面了解操作系统的基本概念和内部工作原理。 该书的第一部分介绍了操作系统的基本概念,例如进程、线程和调度算法。它解释了多任务处理的基本原理,并讨论了与并发性和并行性相关的问题。 第二部分讨论了内存管理和虚拟化技术,包括分页、分段、内存分配和页面置换等。它还讨论了操作系统如何处理死锁和内存泄漏等问题。 第三部分则专注于文件系统和存储技术。它解释了文件系统的层次结构和存储管理,以及如何实现文件和目录的访问控制和保护机制。 总的来说,《操作系统:三步轻松学》这本书以易于理解和深入浅出的方式,介绍了操作系统的关键概念和原理。它旨在帮助读者建立对操作系统的基本了解,以及如何设计和实现一个高效稳定的操作系统。无论是计算机科学专业的学生、软件开发人员还是系统管理员,都可以从这本书中获得宝贵的知识和见解,以提升对操作系统的理解和应用能力。 ### 回答2: 《操作系统:三个简单的组成部分》是一本关于操作系统的英文书籍。该书由Remzi H. Arpaci-Dusseau和Andrea C. Arpaci-Dusseau合著。这本书旨在以简洁易懂的方式介绍操作系统的基本概念和原理。 书中主要涵盖操作系统的三个组成部分,即虚拟化、并发性和持久性。虚拟化是指操作系统利用物理资源来创建多个虚拟资源的能力,例如虚拟内存,使得多个程序可以在同一台机器上同时运行。并发性涉及操作系统如何管理和调度同时运行的多个任务。持久性是指操作系统如何管理和访问长期存储,例如硬盘或闪存。 书中通过详细的示例和清晰的解释,阐述了各种操作系统的基本概念和常见问题的解决方案。它还介绍了操作系统的设计原则和优化技术,以及一些常见的操作系统功能,如进程管理、内存管理和文件系统。 这本书是为初学者和其他对操作系统感兴趣的人编写的,而且不需要太多的预备知识。它的目标是帮助读者理解操作系统的基本原理和如何实现它们。 总之,《操作系统:三个简单的组成部分》是一本很好的操作系统入门书籍,它以易读易懂的方式介绍了操作系统的基本概念和原理。无论是对于学生、软件工程师还是对操作系统感兴趣的人来说,这本书都会提供有价值的知识和见解。 ### 回答3: 《操作系统:三部简单的作品》是一本关于操作系统的英文书籍。这本书由Remzi H. Arpaci-Dusseau和Andrea C. Arpaci-Dusseau撰写,旨在介绍操作系统的基本原理和概念。 该书主要分为三个部分,每个部分都涵盖一个关键主题。第一部分介绍了操作系统的概述和基本概念,包括进程、线程、调度、同步和死锁等。它通过一系列易懂的例子和实验帮助读者理解这些概念,并展示它们在实际操作系统中的应用。 第二部分深入讨论了操作系统的内存管理和虚拟内存的原理。它解释了内存分配、地址转换和页面置换等重要的内存管理技术,以及虚拟内存的概念和工作原理。读者可以通过实践性的编程练习来巩固所学知识。 最后一部分探讨了文件系统和存储技术。它介绍了文件系统的组织和功能,包括文件的访问权限、目录结构和磁盘调度算法。此外,它还讨论了存储技术的发展,例如RAID和闪存。 《操作系统:三部简单的作品》以其简洁明了的风格和通俗易懂的示例而闻名。它适用于那些对操作系统感兴趣的初学者和教育者。读者无需具备深厚的计算机科学背景,即可轻松理解书中的内容。 总之,这本书是一本介绍操作系统基本原理和概念的优秀教材。通过阅读《操作系统:三部简单的作品》,读者可以建立起对操作系统的深入理解,从而在实践中更好地应用和开发操作系统。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值