linux内核完全剖析 学习笔记 打字太累 截图 持续更新

  • 1. linux0.11开机时间

#define MINUTE 60

#define HOUR (60*MINUTE)

#define DAY (24*HOUR)

#define YEAR (365*DAY)

/* interestingly, we assume leap-years */

static int month[12] = {

0,

DAY*(31),

DAY*(31+29),

DAY*(31+29+31),

DAY*(31+29+31+30),

DAY*(31+29+31+30+31),

DAY*(31+29+31+30+31+30),

DAY*(31+29+31+30+31+30+31),

DAY*(31+29+31+30+31+30+31+31),

DAY*(31+29+31+30+31+30+31+31+30),

DAY*(31+29+31+30+31+30+31+31+30+31),

DAY*(31+29+31+30+31+30+31+31+30+31+30)

};

  • 2.结构体赋值

详情百度http://blog.csdn.net/hazir/article/details/9429017

  • 3. 关于do_signal *(&eip) = sa_handler

(1)

http://www.oldlinux.org/oldlinux/viewthread.php?tid=14839&extra=page%3D2

*(&eip) = sa_handler 与 eip = sa_handler 有什么不一样?

网上有说 是因为 两个的类型不一样 一个long 一个 unsigned long ,感觉这不是原因。

应该是另一种说法 eip = sa_handler 这条语句可能导致 编译器进行优化 而栈中的值没有改变  , 应该是这样。

(2)

  • 4.任务切换

ljmp tss段选择符这样的语句会造成任务切换,cpu自动保存当前任务tss内容,并恢复需要切换到的任务的tss内容,造成任务切换。任务1和任务2的描述内容与任务0类似。

  • 5.写时复制

为了介于物理内存,在调用fork()生成新进程时,新进程与原进程会共享同一内存区。只有当其中一个进程进行写操作,系统才会为其另外分配内存页面。这就是写时复制的概念。

  • 6.需求加载(load on demand)/需求分页(demand-paging)

  • 7.指针的指针

  • 8.   I/O端口和寻址

SPARC为统一编址,ld,st指令访问

  • 9.  接口访问控制

  • 10.  C程序编译和链接

  • 11. volatile

  • 12. 嵌入汇编

这里不同处理器用到的寄存器名字应该不一样吧。。。目前看到的资料都是对应x86的eax,ebx啥的。。。然而我要做的事sparc

  • 13.  圆括号中的组合语句

  • 14. 目标文件符号表和字符串部分

  • 15. 连接程序预定义变量

ARINC 653项目遇到过。。。自己有定义了个end变量,值总是不对。。。原来是链接程序预定义的

  • 16. 高尾端/低尾端

不过给我启发的是,在裘宗燕翻译的《程序设计实践》里,这对术语并没有翻译为“大端”和小端,而是“高尾端”和“低尾端”,这就好理解了:如果把一个数看成一个字符串,比如11223344看成"11223344",末尾是个‘\0‘,‘11‘到‘44‘个占用一个存储单元,那么它的尾端(低字节)很显然是44,前面的高还是低就表示尾端放在高地址还是低地址,它在内存中的放法非常直观,如下图:

X86小端,SPARC大端

  • 内存地址空间

  • 17. 分段机制

  • 18. 一致/非一致代码段

一致代码段:

简单理解,就是操作系统拿出来被共享的代码段,可以被低特权级的用户直接调用访问的代码.

通常这些共享代码,是"不访问"受保护的资源和某些类型异常处理。比如一些数学计算函数库,为纯粹的数学运算计算,被作为一致代码段.

一致代码段的限制作用

1.特权级高的程序不允许访问特权级低的数据:核心态不允许调用用户态的数据.

2.特权级低的程序可以访问到特权级高的数据.但是特权级不会改变:用户态还是用户态.

非一致代码段:

为了避免低特权级的访问而被操作系统保护起来的系统代码.

非一致代码段的限制作用

1.只允许同级间访问.

2.绝对禁止不同级访问:核心态不用用户态.用户态也不使用核心态.

通常低特权代码必须通过"门"来实现对高特权代码的访问和调用.

不同级别代码段之间转移规则,是通过CPL/RPL/DPL来校验.

  • 19. sleep_on等待队列

  • 20.定时器链表

  • 21.信号

  • 22 . 块设备

  • 23. extern inline

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-09 16:13:54

linux内核完全剖析 学习笔记 打字太累 截图 持续更新的相关文章

《linux 内核完全剖析》 笔记 由逻辑地址转换成线性地址代码分析

一开始由这段代码引发的纠结 get_base(current->ldt[1]) 下面是各个相关的代码,摘自不同的header files... current是指向当前task的指针 struct desc_struct ldt[3]; struct desc_struct { unsigned long a,b; } ; #define _get_base(addr) ({unsigned long __base; __asm__("movb %3,%%dh\n\t" &quo

《linux 内核完全剖析》 笔记 CODE_SPACE 宏定义分析

在memory.c里面,遇到一个宏定义,如下: #define CODE_SPACE(addr) ((((addr)+4095)&~4095) < current->start_code + current->end_code) 看的第一眼,不知道,第二眼,还是不知道,纠结了半天还是不知道. 睡了一晚,今天早上再看,嘿嘿,居然看懂了... 这个宏定义用于判断给定的addr线性地址是否位于当前进程的代码段中. 4095 = 0xFFF; addr+4095的作用是将位于0~4095

《linux 内核全然剖析》 笔记 CODE_SPACE 宏定义分析

在memory.c里面.遇到一个宏定义,例如以下: #define CODE_SPACE(addr) ((((addr)+4095)&~4095) < current->start_code + current->end_code) 看的第一眼,不知道.第二眼.还是不知道.纠结了半天还是不知道. 睡了一晚,今天早上再看,嘿嘿,竟然看懂了... 这个宏定义用于推断给定的addr线性地址是否位于当前进程的代码段中. 4095 = 0xFFF; addr+4095的作用是将位于0~40

Hadoop学习笔记系列文章导游【持续更新中...】

一.为何要学习Hadoop? 这是一个信息爆炸的时代.经过数十年的积累,很多企业都聚集了大量的数据.这些数据也是企业的核心财富之一,怎样从累积的数据里寻找价值,变废为宝炼数成金成为当务之急.但数据增长的速度往往比cpu和内存性能增长的速度还要快得多.要处理海量数据,如果求助于昂贵的专用主机甚至超级计算机,成本无疑很高,有时即使是保存数据,也需要面对高成本的问题,因为具有海量数据容量的存储设备,价格往往也是天文数字.成本和IT能力成为了海量数据分析的主要瓶颈. Hadoop这个开源产品的出现,打破

ElasticSearch学习笔记--一些规范,会持续更新

我们在ElasticSearch中存储的数据一般是采用json的格式存储,所以ElasticSearch中有一个叫Mapper的东西用来定义jsonschema来规范这个json 但是这个mapper在初期可能会经常变更,比如添加一个字段之类的 但是添加字段的时候就需要重建index,这个时候肯定会对使用者带来很大的问题,可能就直接挂了 所以针对这种情况呢,有一种方案还是不错的: 就是使用别名alias,我们建立索引的时候都给他建立一个别名,我们查询的时候都通过这个别名来访问,然后当我们需要re

《linux 内核完全剖析》 exit.c 代码分析笔记

exit.c 代码分析笔记 release 释放进程的函数release() 主要根据指定进程的任务数据结构指针,在任务数组中删除指定的进程指针,释放相关内存页,并立刻让内核重新调度进程的运行. void release(struct task_struct * p) //释放p指向的进程 { int i; if (!p) //常规检测p是否为0 return; if (p == current) { //不能把自己给释放了 printk("task releasing itself\n\r&q

《linux 内核完全剖析》 fork.c 代码分析笔记

fork.c 代码分析笔记 verifiy_area long last_pid=0; //全局变量,用来记录目前最大的pid数值 void verify_area(void * addr,int size) // addr 是虚拟地址 ,size是需要写入的字节大小 { unsigned long start; start = (unsigned long) addr; //把地址强制类型转换之后,赋值给start size += start & 0xfff; //取addr在当前虚拟地址中4

《linux 内核完全剖析》sched.c sched.h 代码分析笔记

sched.c sched.h 代码分析笔记 首先上header file sched.h #ifndef _SCHED_H #define _SCHED_H #define HZ 100 #define NR_TASKS 64 #define TASK_SIZE 0x04000000 #define LIBRARY_SIZE 0x00400000 #if (TASK_SIZE & 0x3fffff) #error "TASK_SIZE must be multiple of 4M&qu

《linux 内核完全剖析》 signal.c 代码分析笔记

signal.c 代码分析笔记 sgetmask int sys_sgetmask()// 获取当前进程阻塞的信号 { returncurrent->blocked; } sys_ssetmask int sys_ssetmask(int newmask) //设置当前进程阻塞信号,确保SIGKILL 和SIGSTOP不被阻塞 { int old=current->blocked; current->blocked= newmask & ~(1<<(SIGKILL-1