[关闭]
@SovietPower 2021-10-24T15:10:50.000000Z 字数 12121 阅读 1960

OS 实验2 创建共享内存 thread_create()分析

OS



https://www.zybuluo.com/SovietPower/note/1824664


创建共享内存

通过共享内存,完成生产者-消费者模型的创建,并用gdb进行调试。

使用man+函数名即可查看函数帮助文档。

mmap()

  1. #include <sys/mman.h>
  2. void *mmap(void *start, size_t length, int prot, int flags, int fd, off_t offset);
  3. int munmap(void *start, size_t length);

创建新的虚拟内存区域,并将一个文件或对象映射到该区域。
执行成功时,返回被映射区域的指针;否则返回MAP_FAILED,其值为(void *)-1

参数:
start:映射区的开始地址,为0时表示由系统决定映射区的起始地址。
length:映射区的长度,以字节为单位(会补齐到整数倍内存页大小)。
prot:期望的内存保护标志(使用该空间的权限),不能与文件的打开模式冲突。

prot可通过or组合以下值作为参数:
PROT_EXEC:页内容可被执行。
PROT_READ:页内容可读。
PROT_WRITE:页内容可写。
PROT_NONE:页不可访问。

flags:指定映射对象的类型,映射选项和映射页是否可以共享。

flags可通过or组合以下值作为参数(仅列举常用值):
MAP_FIXED:使用指定的映射起始地址,且起始地址必须落在页的边界上。如果由start和len参数指定的内存区与已有的映射空间重叠,重叠部分会被丢弃。如果指定的起始地址不可用,操作失败。
MAP_SHARED:与其它所有映射这个对象的进程共享映射空间。对共享区的写入,相当于输出到文件。但进程在映射空间对共享内容的改变并不会直接写到磁盘文件中,只有调用msync()munmap()后,共享区才会被更新。与MAP_PRIVATE互斥,只能且必须使用其中一个。
MAP_PRIVATE:建立一个写时拷贝的私有映射。对内存区域的写入不会影响到原文件。与MAP_SHARED互斥,只能且必须使用其中一个。
MAP_ANONYMOUS:匿名映射,映射区不与任何文件关联(无需指定fd)。可避免文件的创建与打开,但只能用于具有父子关系的进程。

fd:文件描述符,一般为open()的返回值。可以为-1,表示不指定文件,此时flags必须包含MAP_ANON
offset:被映射对象内容的起点。

功能:
1. 允许用户程序直接访问设备内存,相比于在用户空间和内核空间互相拷贝数据,效率更高。
2. 使进程之间可通过映射同一个普通文件实现共享内存。

munmap()

  1. #include <sys/mman.h>
  2. void *mmap(void *start, size_t length, int prot, int flags, int fd, off_t offset);
  3. int munmap(void *start, size_t length);

取消start所指的虚拟映射内存,length表示取消的空间大小。
执行成功时,返回0;否则返回-1。

进程结束,或通过exec执行其他程序时,映射内存会自动被解除。但关闭文件描述符时不会解除映射。

shm_open()

  1. #include <fcntl.h> /* For O-* constants */
  2. #include <sys/stat.h> /* For mode constants */
  3. #include <sys/mman.h>
  4. int shm_open(const char *name, int oflag, mode_t mode);
  5. int shm_unlink(const char *name);

注意编译时要链接库,即在最后加参数-lrt
shm_open的帮助文档中,语法要求Link with -lrt.。NOTES中:Programs using these functions must specify the -lrt flag to cc in order to link against the required ("realtime") library.

Linux共享内存通过tmpfs文件系统实现。tmpfs文件系统完全驻留在RAM中,其读写速度极快。
tmpfs默认位于/dev/shm目录,因此对该目录下的文件读写即通过tmpfs文件系统进行,其速度与读写内存速度一样。
/dev/shm的默认容量为系统内存的一半。只有在其中含有文件时,才真正占用对应的内存大小,否则不会占用内存。

用于打开或创建文件。
执行成功时,返回对应文件描述符;否则返回-1。返回的文件描述符一定是最小的未被使用的描述符。

shm_openopen基本相同,但其操作的文件一定位于tmpfs文件系统,即位于/dev/shm

参数:
name:指定要打开或创建的文件名。注意因为shm_open操作的文件位于/dev/shm,所以不需且不能包含路径,不同于open()pathname(不过也可包含路径,但要保证/dev/shm中包含对应路径。此外tmpfs不是一定在/dev/shm中)。
oflag:指定打开或创建的文件模式。

oflags可通过or组合以下值作为参数(仅列举常用值):
O_RDONLY:以只读模式打开。
O_WRONLY:以只写模式打开。
O_RDWR:以可读可写模式打开。以上三种模式只可选择一种。
O_APPEND:以追加方式打开。
O_CREAT:如果文件不存在,则创建文件。
O_EXCL:如果使用了O_CREAT且文件已存在,返回-1(并更新错误信息errno)。
O_TRUNC:如果文件已存在(且以可写模式打开),清空该文件。

mode:使用O_CREAT新建文件时,该文件的权限标志。由4位数字组成。

mode的第一位数代表特殊权限(suid:4,sgid:2,sbit:1,即setUid/setGid/粘着位),一般为0即可,可省略。
后三位数分别表示:所有者、群组、其他用户所具有的权限。每位数通过加权表示权限:4:读权限,2:写权限,1:执行权限。

  1. #include <fcntl.h> /* For O-* constants */
  2. #include <sys/stat.h> /* For mode constants */
  3. #include <sys/mman.h>
  4. int shm_open(const char *name, int oflag, mode_t mode);
  5. int shm_unlink(const char *name);

删除/dev/shm目录下的指定文件。
执行成功时,返回0;否则返回-1。

unlink()并指定/dev/shm+name作为目录,可实现同样效果。但tmpfs不是一定在/dev/shm中。
shm_open()创建的文件,如果不使用shm_unlink()删除,会一直位于/dev/shm中,直到操作系统重启或用rm删除。

ftruncate()

  1. #include <unistd.h>
  2. #include <sys/types.h>
  3. int truncate(const char *path, off_t length);
  4. int ftruncate(int fd, off_t length);

truncateftruncate均可重置文件大小为length字节。若文件缩小,则部分信息会丢失;若文件扩大,则用空字符\0填充。
执行成功时,返回0;否则返回-1。

任何通过open()shm_open()打开的文件都可使用。
使用truncate,需保证文件可写;使用ftruncate,需保证文件已打开且可写。

生产者-消费者模型调试

编译文件
文件使用传入的参数决定运行生产者还是消费者。

调试生产者、运行消费者
在输出信息的位置(78行)添加断点。使用producer参数运行生产者。
使用consumer参数运行消费者。

输出/接收信息
调试在输出信息的位置(断点处)暂停,生产者用continue输出一条信息,消费者收到信息。

输出/接收信息(第二条)
调试在输出第二条信息的位置暂停,生产者继续用continue输出第二条信息,消费者收到第二条信息。

输出/接收信息(第三条)
调试在输出第三条信息的位置暂停,生产者继续用continue输出第三条信息,消费者收到第三条信息。

代码:
使用需传入参数:pproducer表示生产者,cconsumer表示消费者。

  1. #include <cstdio>
  2. #include <cstdlib>
  3. #include <cstring>
  4. #include <fcntl.h>
  5. #include <unistd.h>
  6. #include <sys/mman.h>
  7. #include <sys/stat.h>
  8. void *Mmap(void *start, size_t length, int prot, int flags, int fd, off_t offset)
  9. {
  10. void *ptr = mmap(start, length, prot, flags, fd, offset);
  11. if(ptr==MAP_FAILED)
  12. {
  13. puts("mmap failed.");
  14. exit(-1);
  15. }
  16. return ptr;
  17. }
  18. int Munmap(void *start, size_t length)
  19. {
  20. int res=munmap(start, length);
  21. if(res==-1)
  22. {
  23. puts("munmap failed.");
  24. exit(-1);
  25. }
  26. return res;
  27. }
  28. int Shm_open(const char *name, int oflag, mode_t mode)
  29. {
  30. int res=shm_open(name, oflag, mode);
  31. if(res==-1)
  32. {
  33. puts("shm_open failed.");
  34. exit(-1);
  35. }
  36. return res;
  37. }
  38. int Shm_unlink(const char *name)
  39. {
  40. int res=shm_unlink(name);
  41. if(res==-1)
  42. puts("shm_unlink failed."), exit(-1);
  43. return res;
  44. }
  45. int Ftruncate(int fd, off_t length)
  46. {
  47. int res=ftruncate(fd, length);
  48. if(res==-1)
  49. puts("ftruncate failed."), exit(-1);
  50. return res;
  51. }
  52. const int SIZE = 4096;
  53. const char *NAME = "Messages";
  54. namespace Producer
  55. {
  56. const int message_size=3;
  57. const char *messages[message_size]={
  58. "message 1,",
  59. "message 2,",
  60. "message 3!"
  61. };
  62. int main()
  63. {
  64. puts("Producer begins.");
  65. //用tmpfs文件系统创建文件,并设置大小
  66. int shm_fd = Shm_open(NAME, O_RDWR|O_CREAT, 0666);
  67. Ftruncate(shm_fd, SIZE);
  68. //将文件映射到ptr所指的虚拟内存区域
  69. void *ptr = Mmap(0, SIZE, PROT_WRITE, MAP_SHARED, shm_fd, 0);
  70. for(int i=0; i<message_size; ++i)
  71. {
  72. sprintf((char *)ptr, "%s", messages[i]);
  73. ptr += strlen(messages[i]);
  74. }
  75. return 0;
  76. }
  77. }
  78. namespace Consumer
  79. {
  80. int main()
  81. {
  82. puts("Consumer begins.");
  83. //打开与生产者相同的文件
  84. int shm_fd = Shm_open(NAME, O_RDONLY, 0666);
  85. //将同一文件映射到虚拟内存区域,以共享内存
  86. void *ptr = Mmap(0, SIZE, PROT_READ, MAP_SHARED, shm_fd, 0);
  87. for(int i=0; i<3; )
  88. if(strlen((char *)ptr)>0)
  89. {
  90. printf("Consumer: %s\n", (char *)ptr);
  91. ptr += strlen((char *)ptr);
  92. ++i; //强制接收3条消息
  93. }
  94. //消费者使用完成后删除文件(shm_open()创建的文件不会在结束时自动删除)
  95. Shm_unlink(NAME);
  96. return 0;
  97. }
  98. }
  99. int main(int argc, char **argv)
  100. {
  101. for(int i=0; i<argc; ++i)
  102. printf("argv[%d]=%s\n",i,argv[i]);
  103. if(argc!=2)
  104. return puts("Argument Error."), 1;
  105. if(argv[1][0]=='p')
  106. return Producer::main();
  107. else if(argv[1][0]=='c')
  108. return Consumer::main();
  109. else
  110. return puts("Argument Error."), 1;
  111. return 0;
  112. }

thread_create()解析

分析pintos中创建线程的函数thread_create()的源代码。

函数原型:

  1. #include <thread.c>
  2. tid_t thread_create (const char *name, int priority, thread_func *, void *);

函数介绍:

宏定义

tid_t
线程标识符,为int。

  1. typedef int tid_t;

uint8_t
8位无符号整数,即unsigned char。用于表示栈指针。

  1. typedef unsigned char uint8_t;

thread_func
void类型函数的函数指针,参数为void *aux

  1. typedef void thread_func (void *aux);

PGBITS
页的位数,为

  1. #define PGBITS 12 /* Number of offset bits. */

PGSIZE
页的大小,为

  1. #define PGSIZE (1 << PGBITS) /* Bytes in a page. */

THREAD_MAGIC
堆栈金丝雀的默认值,为一个随机出的值。

  1. #define THREAD_MAGIC 0xcd6abf4b

链表

内核中使用循环链表。

结构list表示循环链表,含有头、尾两个链表元素。
结构list_elem表示链表元素,含有两个双向的链表元素指针。

相关结构体/函数

allocate_tid()

为新线程分配一个可用的tid。
会执行:将tid_lock上锁(阻塞后来需要分配tid的线程);分配tid,值为next_tid++;将tid_lock解锁(解除后来线程的阻塞)。

tid_lock:进程标识符tid的锁。用于分配tid。

  1. static tid_t allocate_tid (void);

alloc_frame()

为线程t分配size字节的栈帧(将其栈顶指针减少size),并返回当前的栈指针。
size必须为整数倍的字大小(即字节)。

  1. static void *alloc_frame (struct thread *t, size_t size);

intr_level

用来表示是否接受中断。含两种值。
关闭中断是pintos实现原子操作的方式。

  1. enum intr_level
  2. {
  3. INTR_OFF, /* 关闭中断 */
  4. INTR_ON /* 开启中断 */
  5. };

intr_disable()

关闭中断,并返回调用前的中断状态。

  1. enum intr_level intr_disable (void);

通过该函数,可以实现在执行某操作时不被中断,从而实现一个原子操作

  1. enum intr_level old_level = intr_disable ();
  2. // do something without being interrupted
  3. intr_set_level (old_level);

intr_enable()

开启中断,并返回调用前的中断状态。

  1. enum intr_level intr_enable (void);

init_thread()

初始化一个线程t,并将其加入到all_list(一个包含所有线程的链表)。
初始化包括:将其命名为name,设置其优先级为priority;设置状态status为阻塞THREAD_BLOCKED,调用enum intr_level intr_disable (void);关闭中断,并更新old_levelINTR_OFF(?);设置栈指针(值为(uint8_t *) t + PGSIZE);设置金丝雀值magic为THREAD_MAGIC

  1. static void init_thread (struct thread *t, const char *name, int priority);

kernel_thread()

内核线程的基础函数。
用于:开启中断接收(调度程序在运行时关闭了中断接收),执行线程的函数,结束并杀死线程。

  1. static void
  2. kernel_thread (thread_func *function, void *aux)
  3. {
  4. ASSERT (function != NULL);
  5. intr_enable (); /* 调度程序在运行时会关闭中断 */
  6. function (aux); /* 执行线程函数 */
  7. thread_exit (); /* 如果函数成功返回,结束线程 */
  8. }

kernel_thread_frame

内核线程函数kernel_thread()的栈帧。
保存了:线程函数的返回地址、要调用的函数、调用函数的辅助信息。

  1. struct kernel_thread_frame
  2. {
  3. void *eip; /* 返回地址 */
  4. thread_func *function; /* 要调用的函数 */
  5. void *aux; /* 函数的辅助信息 */
  6. };

lock

锁。
包含两个值:拥有该锁的线程(用于调试),控制权限的二进制信号量。

  1. struct lock
  2. {
  3. struct thread *holder; /* Thread holding lock (for debugging). */
  4. struct semaphore semaphore; /* Binary semaphore controlling access. */
  5. };

lock_acquire()

请求锁。
会将给定锁的信号量减1(如果减1前为0则等待,直至其为正可减),并设置给定锁的所有者为当前线程thread_current()
该函数可能会等待(sleep),所以不能在中断处理程序中被调用(会导致内核错误)。

  1. void lock_acquire (struct lock *lock);

lock_release()

释放锁。
会将给定锁的信号量加1,并设置给定锁的所有者为NULL。
需保证锁被当前线程拥有。

  1. void lock_release (struct lock *lock);

palloc_flags

表示分配的页的模式。
包含三种:PAL_ASSERT(内核错误),PAL_ZERO(将页用0填充),PAL_USER(用户页,表示从用户池中获取页,否则从内核池中获取页)。

  1. enum palloc_flags
  2. {
  3. PAL_ASSERT = 001, /* 内核错误 */
  4. PAL_ZERO = 002, /* 页内容用0填充 */
  5. PAL_USER = 004 /* 用户页 */
  6. };

内核错误(Kernel Panic)指操作系统在监测到内部的致命错误,但无法安全处理此错误时采取的操作。此时内核会尽可能将它此时能获取的全部信息打印出来。
常见原因
1. 中断处理程序执行时,它不处于任何一个进程上下文,此时使用可能导致睡眠的函数(如信号量),会破坏系统调度,导致内核错误。
2. 栈溢出。
3. 对于除0异常、内存访问越界、缓冲区溢出等错误,若发生在应用程序,则内核的异常处理程序会进行处理,即使终止原程序也不会影响其它程序;若发生在内核,则会引起内核错误。
4. 内核陷入死锁状态,自旋锁有嵌套使用的情况。
5. 内核线程中存在死循环。

palloc_get_page()

创建一个页,返回其虚拟地址(如果无可用页则返回NULL)。
新页的模式与传入的flags有关:PAL_ASSERT(内核错误),PAL_ZERO(将页用0填充),PAL_USER(用户页,表示从用户池中获取页,否则从内核池中获取页)。

  1. void *palloc_get_page (enum palloc_flags flags);

switch_entry()

进程切换入口?switch.c中无该函数的源码,在switch.S中有。

  1. void switch_entry (void);

switch_entry_frame

switch_entry()的栈帧。
保存了一个返回地址,其类型为void类型函数的指针。

  1. struct switch_entry_frame
  2. {
  3. void (*eip) (void);
  4. };

switch_threads()

进程切换函数。switch.c中无该函数的源码,在switch.S中有。
将当前进程从cur切换到next:保存next的上下文,切换到cur的上下文。
cur必须是当前正在运行的线程,next必须正在运行该函数。

  1. struct thread *switch_threads (struct thread *cur, struct thread *next);

switch_threads_frame

switch_threads()的栈帧。
保存了4个寄存器的值、返回地址(为void类型函数的指针)、switch_threads()的cur参数、switch_threads()的next参数。

  1. struct switch_threads_frame
  2. {
  3. uint32_t edi; /* 0: 保存 %edi */
  4. uint32_t esi; /* 4: 保存 %esi */
  5. uint32_t ebp; /* 8: 保存 %ebp */
  6. uint32_t ebx; /* 12: 保存 %ebx */
  7. void (*eip) (void); /* 16: 返回地址 */
  8. struct thread *cur; /* 20: switch_threads()的 CUR 参数 */
  9. struct thread *next; /* 24: switch_threads()的 NEXT 参数 */
  10. };

thread

一个线程结构表示一个内核线程或用户进程。
一个线程结构存储在一个4KB的页中。页的最下方(偏移量为0的位置)存储页信息,通常为若干字节,不超过1KB;页的剩余部分为内核栈,自顶向下增长(偏移量为4KB的位置)。
页信息的最上方为magic,即栈内金丝雀,用以判断栈使用的空间是否过大。
elem是一个链表元素,既可以表示运行队列里的一个元素(当线程处于就绪状态时,位于thread.c),也可表示信号等待队列里的一个元素(当线程处于阻塞状态时,位于synch.c)。
每个线程只含有4KB的栈大小,所以大数组或大的数据结构应动态分配其内存。

  1. struct thread
  2. {
  3. /* Owned by thread.c. */
  4. tid_t tid; /* 线程标识符 */
  5. enum thread_status status; /* 线程状态 */
  6. char name[16]; /* 线程名称(调试用) */
  7. uint8_t *stack; /* 栈指针 */
  8. int priority; /* 优先级 */
  9. struct list_elem allelem; /* 链表元素,用于放在一个包含所有线程链表中 */
  10. /* 该元素在 thread.c 和 synch.c 间共享 */
  11. struct list_elem elem; /* 链表元素 */
  12. #ifdef USERPROG
  13. /* Owned by userprog/process.c. */
  14. uint32_t *pagedir; /* 页路径(当线程为用户进程时) */
  15. #endif
  16. /* Owned by thread.c. */
  17. unsigned magic; /* 检测栈溢出 */
  18. };

thread_current()

返回正在运行的线程。此外会对要返回的结果进行检查。

  1. struct thread *thread_current (void);

thread_start()

执行需要抢先执行的线程(preemptive thread,由中断安排)。同时创建idle线程。

  1. void thread_start (void);

thread_status

表示进程所处的状态。

  1. enum thread_status
  2. {
  3. THREAD_RUNNING, /* Running thread. */
  4. THREAD_READY, /* Not running but ready to run. */
  5. THREAD_BLOCKED, /* Waiting for an event to trigger. */
  6. THREAD_DYING /* About to be destroyed. */
  7. };

thread_block()

将当前进程转为阻塞状态。这样阻塞的进程在调用thread_unblock()之前不会再被调度。
必须在关闭中断时使用。
阻塞当前进程后,会调用schedule()(见实验3)调度其它可用进程。

  1. void thread_block (void);

thread_unblock()

将进程t从阻塞状态转为就绪状态,并将其加入到就绪队列。
需保证t处于阻塞状态。
该函数不会取代正在运行的线程。如果调用者关闭了中断,它可能会自动解除一个线程的阻塞并更新其它信息(?)。

  1. void thread_unblock (struct thread *t);

解析

功能:
创建一个内核线程,其名称为name,优先级为priority,需要执行函数function,需要的参数为aux。
如果创建成功,返回其tid;否则返回TID_ERROR

整体过程:
1. 为新线程创建一个页、分配tid,并将其设为阻塞状态。
2. 创建三个函数的栈帧,这三个函数用于切换并调用线程。(猜测)当线程被调度执行时,调用线程切换函数switch_threads(),然后进入相应的线程切换入口switch_entry(),最后进入函数kernel_thread(),执行该线程的函数、最终杀死线程。
3. 创建完栈帧后,将进程设为就绪状态。
4. 返回其tid。

注意:
如果thread_start()已经被调用,则在thread_create()返回之前,新线程可能就已被安排调用,甚至已经结束返回。相反地,在新线程被安排调用之前,原线程可能会运行任意长的时间。如果要确保有序,需使用信号量或其他的同步方式。
该函数为新线程分配了优先级priority,但pintos利用优先级影响调度的功能还未实现,这就是问题1-3的目标。

代码:

  1. //所有涉及的类型与函数均在上面介绍过
  2. tid_t
  3. thread_create (const char *name, int priority,
  4. thread_func *function, void *aux)
  5. {
  6. // 定义新线程的指针t,为函数 kernel_thread(), switch_entry(), switch_threads() 分配栈帧。
  7. // 栈帧用于保存函数的信息/数据。
  8. struct thread *t;
  9. struct kernel_thread_frame *kf;
  10. struct switch_entry_frame *ef;
  11. struct switch_threads_frame *sf;
  12. tid_t tid;
  13. ASSERT (function != NULL);
  14. // 为线程分配1页,令t指向分配的空间。若无可用页则返回TID_ERROR。
  15. t = palloc_get_page (PAL_ZERO);
  16. if (t == NULL)
  17. return TID_ERROR;
  18. // 初始化线程(具体内容见`init_thread()`),并分配其tid。
  19. init_thread (t, name, priority);
  20. tid = t->tid = allocate_tid ();
  21. // 分配`kernel_thread()`的栈帧,初始化信息为:返回地址NULL;要调用的函数function;参数aux。
  22. kf = alloc_frame (t, sizeof *kf);
  23. kf->eip = NULL;
  24. kf->function = function;
  25. kf->aux = aux;
  26. // 分配`switch_entry()`的栈帧,初始化信息为:返回地址,指向`kernel_thread()`。
  27. ef = alloc_frame (t, sizeof *ef);
  28. ef->eip = (void (*) (void)) kernel_thread;
  29. // 分配`switch_threads()`的栈帧,初始化信息为:返回地址,指向`switch_threads()`;保存的%ebp值为0。
  30. sf = alloc_frame (t, sizeof *sf);
  31. sf->eip = switch_entry;
  32. sf->ebp = 0;
  33. // 解除线程t的阻塞状态,设为就绪状态,并将其加入到就绪队列。
  34. thread_unblock (t);
  35. // 创建成功,返回其tid。
  36. return tid;
  37. }
添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注