Skip to content

Commit 73d9215

Browse files
committed
Add ucontext.md
1 parent 55e846c commit 73d9215

File tree

1 file changed

+148
-0
lines changed

1 file changed

+148
-0
lines changed

context/ucontext.md

+148
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,148 @@
1+
## ucontext
2+
3+
`ucontext.h` or `x86_64-linux-gnu/sys/ucontext.h`
4+
5+
```c
6+
/* Structure to describe FPU registers. */
7+
typedef struct _libc_fpstate *fpregset_t;
8+
9+
/* Context to describe whole processor state. */
10+
typedef struct
11+
{
12+
gregset_t gregs;
13+
/* Note that fpregs is a pointer. */
14+
fpregset_t fpregs;
15+
__extension__ unsigned long long __reserved1 [8];
16+
} mcontext_t;
17+
18+
/* Userlevel context. */
19+
typedef struct ucontext
20+
{
21+
unsigned long int uc_flags;
22+
struct ucontext *uc_link;
23+
stack_t uc_stack;
24+
mcontext_t uc_mcontext;
25+
__sigset_t uc_sigmask;
26+
struct _libc_fpstate __fpregs_mem;
27+
} ucontext_t
28+
29+
/* Definitions taken from the kernel headers. */
30+
struct _libc_fpreg
31+
{
32+
unsigned short int significand[4];
33+
unsigned short int exponent;
34+
};
35+
36+
struct _libc_fpstate
37+
{
38+
unsigned long int cw;
39+
unsigned long int sw;
40+
unsigned long int tag;
41+
unsigned long int ipoff;
42+
unsigned long int cssel;
43+
unsigned long int dataoff;
44+
unsigned long int datasel;
45+
struct _libc_fpreg _st[8];
46+
unsigned long int status;
47+
};
48+
49+
int getcontext(ucontext_t* ucp);
50+
int setcontext(const ucontext_t* ucp);
51+
void makecontext(ucontext_t* ucp, void (*func)(), int argc, ...);
52+
int swapcontext(ucontext_t* ocp, const ucontext_t* ucp);
53+
```
54+
55+
* `ucontext_t`
56+
57+
其中 `uc_stack` 是上下文切换时使用的栈空间,`uc_link` 指向下一个上下文。
58+
59+
60+
* `getcontext` & `setcontext`
61+
调用 `getcontext` 把当前的上下文保存到 `ucontext_t` 结构中,`setcontext` 将保存的上下文信息设为程序当前的上下文。
62+
下面这个程序会一直不停的打印"hello world"。
63+
64+
```c
65+
#include <ucontext.h>
66+
#include <stdio.h>
67+
68+
int main()
69+
{
70+
ucontext_t context;
71+
getcontext(&context);
72+
printf("hello world\n");
73+
setcontext(&context);
74+
75+
return 0;
76+
}
77+
```
78+
79+
* `makecontext`
80+
81+
自定义一个上下文,它的第二个参数是指向要调用函数的指针,第三个参数是调用该函数时参数的个数,之后的可变参数是调用函数时的参数。
82+
下面这个程序只会打印 "I'm parameter.",不会打印 "return"。
83+
```c
84+
#include <ucontext.h>
85+
#include <stdio.h>
86+
87+
unsigned char stack[10240];
88+
89+
int func(const char* str)
90+
{
91+
printf("%s\n",str);
92+
return 0;
93+
}
94+
95+
int main()
96+
{
97+
ucontext_t context;
98+
getcontext(&context);
99+
context.uc_stack.ss_sp = stack;
100+
context.uc_stack.ss_size = 10240;
101+
context.uc_link = NULL;
102+
makecontext(&context, (void(*)(void)) func , 1,"I'm parameter.");
103+
setcontext(&context);
104+
printf("return\n");
105+
return 0;
106+
}
107+
```
108+
109+
* `swapcontext`
110+
111+
保存当前上下文到第一个参数中,然后将第二个参数的内容设为当前的上下文。
112+
下面这个程序,`uc_link` 设为back,使用 `swapcontext`保存当前的上下文到`back`中,因而能够打印 "return"。
113+
```c
114+
#include <ucontext.h>
115+
#include <stdio.h>
116+
117+
unsigned char stack[10240];
118+
119+
int func(const char* str)
120+
{
121+
printf("%s\n",str);
122+
return 0;
123+
}
124+
125+
int main()
126+
{
127+
ucontext_t context,back;
128+
getcontext(&context);
129+
context.uc_stack.ss_sp = stack;
130+
context.uc_stack.ss_size = 10240;
131+
context.uc_link = &back;
132+
makecontext(&context, (void(*)(void)) func , 1,"I'm parameter.");
133+
swapcontext(&back, &context);
134+
printf("return\n");
135+
return 0;
136+
}
137+
```
138+
139+
## 协同程序
140+
141+
142+
协程包含两个操作恢复(resume)和让出(yield),此外在加上创建:
143+
- 创建
144+
将协程的函数和参数保存起来,返回一个指示它们的句柄。
145+
- 恢复
146+
第一次调用,通过 `makecontext``swapcontext` 启动协程函数。
147+
- 让出
148+
通过 `swapcontext` 切换到恢复函数中调用`swapcontext`的位置。

0 commit comments

Comments
 (0)