- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我决定我应该尝试实现协程(我认为我应该这样称呼它们)以获得乐趣和利润。我希望必须使用汇编程序,如果我想让它对任何事情真正有用,可能还需要一些 C。
请记住,这是出于教育目的。使用已经构建的协程库太容易了(而且真的没有乐趣)。
你们知道setjmp
和 longjmp
?它们允许您将堆栈展开到预定义的位置,并从那里恢复执行。但是,它不能倒带到堆栈上的“稍后”。只能早点回来。
jmpbuf_t checkpoint;
int retval = setjmp(&checkpoint); // returns 0 the first time
/* lots of stuff, lots of calls, ... We're not even in the same frame anymore! */
longjmp(checkpoint, 0xcafebabe); // execution resumes where setjmp is, and now it returns 0xcafebabe instead of 0
longjmp
对另一个。一旦它返回到另一个函数,它必须从它离开的地方恢复(即,在将控制权交给另一个函数的调用期间或之后),有点像
longjmp
返回
setjmp
.
A
创建并行堆栈并将其归零(分配内存和所有这些)。 A
将其所有寄存器压入当前堆栈。 A
将堆栈指针和基指针设置到该新位置,并推送一个神秘的数据结构,指示从哪里跳回以及将指令指针设置回哪里。 A
将其大部分寄存器清零并将指令指针设置为函数 B
的开头. B
在那个堆栈上工作,做任何它需要的工作。 B
到了需要中断并给出 A
的地步再次控制。 B
将所有寄存器压入堆栈,获取神秘的数据结构 A
一开始就给了,并将堆栈指针和指令指针设置为A
告诉它。在这个过程中,它交还A
一个新的、修改过的数据结构,告诉在哪里恢复 B
. A
唤醒,弹出它推送到其堆栈的所有寄存器,并一直工作直到需要中断并给出 B
再次控制。 pusha
将所有寄存器发送到堆栈的指令。然而,处理器架构不断发展,现在有了 x86_64,我们有了更多的通用寄存器,可能还有几个 SSE 寄存器。我找不到任何证据表明 pusha
确实插入了他们。现代 x86 CPU 中大约有 40 个公共(public)寄存器。我必须做所有的push
是我自己吗?而且,没有push
对于 SSE 寄存器(尽管肯定会有一个等效项——我对整个“x86 汇编程序”不熟悉)。 mov rip, rax
(英特尔语法)?此外,从它获取值必须有些特殊,因为它不断变化。如果我喜欢 mov rax, rip
(再次英特尔语法),将 rip
定位于mov
指令,到它之后的指令,还是介于两者之间?只是jmp foo
.假的。 pthread
对于线程? 最佳答案
你是对的 PUSHA
不适用于 x64,它会引发异常 #UD
, 如 PUSHA
只推送 16 位或 32 位通用寄存器。见 Intel manuals获取您想知道的所有信息。
设置 RIP
很简单,jmp rax
将设置 RIP
至 RAX
.要检索 RIP,如果您已经知道所有协程导出源,您可以在编译时获取它,或者您可以在运行时获取它,您可以在该调用之后调用下一个地址。像这样:
a:
call b
b:
pop rax
RAX
现在将是
b
.这是有效的,因为
CALL
压入下一条指令的地址。这种技术也适用于 IA32(虽然我认为在 x64 上有更好的方法,因为它支持 RIP 相对寻址,但我不知道)。当然如果你做一个函数
coroutine_yield
,它可以只是拦截调用者地址:)
A
?那可能没有必要。
coroutine_state
包含以下内容:
initarg
arg
registers
(也包含标志)caller_registers
coroutine_state* coroutine_init(void (*coro_func)(coroutine_state*), void* initarg);
coro_func
是一个指向协程函数体的指针。
coroutine_state
结构 cs
initarg
至 cs.initarg
,这些将是协程 coro_func
至 cs.registers.rip
cs.registers
(不是寄存器,只有标志,因为我们需要一些理智的标志来防止世界末日)cs.registers.rsp
coroutine_state
结构 void* coroutine_next(coroutine_state cs, void* arg)
cs
是从
coroutine_init
返回的结构代表一个协程实例,以及
arg
将在协程恢复执行时被送入协程。
cs.caller_registers
中除了 RSP
,请参阅步骤 3。arg
在 cs.arg
cs.caller_registers.rsp
),添加 2*sizeof(void*)
如果幸运的话会修复它,你必须查看它以确认它,你可能希望这个函数是 stdcall 以便在调用它之前没有寄存器被篡改 mov rax, [rsp]
, 分配 RAX
至 cs.caller_registers.rip
;解释:除非你的编译器是破解版,[RSP]
将持有指向调用此函数的调用指令之后的指令的指令指针(即:返回地址)cs.registers
加载标志和寄存器jmp cs.registers.rip
,有效地恢复协程的执行 coroutine_yield
)。另请注意,在此函数中,您可能会遇到许多复杂情况,例如 C 编译器生成的函数序言和结尾,还可能需要注册参数,您必须处理所有这些问题。就像我说的,stdcall 会为你省去很多麻烦,我认为 gcc 的 -fomit-frame_pointer 会删除结尾的东西。
void coroutine_yield(void* ret);
coroutine_next
的调用者。 .
in cs.registers
cs.registers.rsp
),再次添加 2*sizeof(void*)
到它,并且你希望这个函数也是 stdcall mov rax, arg
(让我们假设编译器中的所有函数都在 RAX
中返回它们的参数) cs.caller_registers
加载标志/寄存器jmp cs.caller_registers.rip
这基本上是从 coroutine_next
返回的调用协程调用者的堆栈帧,并且由于返回值传入 RAX
, 我们返回 arg
.让我们说如果 arg
是 NULL
,则协程已终止,否则为任意数据结构。 coroutine_init
初始化一个协程。 ,那么你可以用
coroutine_next
重复调用实例化的协程.
void my_coro(coroutine_state cs)
cs.initarg
保存初始函数参数(想想构造函数)。每次
my_coro
被称为,
cs.arg
具有由
coroutine_next
指定的不同参数.这就是协程调用者与协程通信的方式。最后,每次协程想要暂停自己时,它都会调用
coroutine_yield
, 并向它传递一个参数,即协程调用程序的返回值。
关于c - 如何创建并行堆栈并在其上运行协程?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3089841/
前言: 有时候,一个数据库有多个帐号,包括数据库管理员,开发人员,运维支撑人员等,可能有很多帐号都有比较大的权限,例如DDL操作权限(创建,修改,删除存储过程,创建,修改,删除表等),账户多了,管理
所以我用 Create React App 创建并设置了一个大型 React 应用程序。最近我们开始使用 Storybook 来处理和创建组件。它很棒。但是,当我们尝试运行或构建应用程序时,我们不断遇
遵循我正在创建的控件的代码片段。这个控件用在不同的地方,变量也不同。 我正在尝试编写指令来清理代码,但在 {{}} 附近插入值时出现解析错误。 刚接触 Angular ,无法确定我错过了什么。请帮忙。
我正在尝试创建一个 image/jpeg jax-rs 提供程序类,它为我的基于 post rest 的 Web 服务创建一个图像。我无法制定请求来测试以下内容,最简单的测试方法是什么? @POST
我一直在 Windows 10 的模拟器中练习 c。后来我改用dev C++ IDE。当我在 C 中使用 FILE 时。创建的文件的名称为 test.txt ,而我给出了其他名称。请帮助解决它。 下面
当我们创建自定义 View 时,我们将 View 文件的所有者设置为自定义类,并使用 initWithFrame 或 initWithCode 对其进行实例化。 当我们创建 customUITable
我正在尝试为函数 * Producer 创建一个线程,但用于创建线程的行显示错误。我为这句话加了星标,但我无法弄清楚它出了什么问题...... #include #include #include
今天在做项目时,遇到了需要创建JavaScript对象的情况。所以Bing了一篇老外写的关于3种创建JavaScript对象的文章,看后跟着打了一遍代码。感觉方法挺好的,在这里与大家分享一下。 &
我正在阅读将查询字符串传递给 Amazon 的 S3 以进行身份验证的文档,但似乎无法理解 StringToSign 的创建和使用方式。我正在寻找一个具体示例来说明 (1) 如何构造 String
前言:我对 C# 中任务的底层实现不太了解,只了解它们的用法。为我在下面屠宰的任何东西道歉: 对于“我怎样才能开始一项任务但不等待它?”这个问题,我找不到一个好的答案。在 C# 中。更具体地说,即使任
我有一个由一些复杂的表达式生成的 ILookup。假设这是按姓氏查找人。 (在我们简单的世界模型中,姓氏在家庭中是唯一的) ILookup families; 现在我有两个对如何构建感兴趣的查询。 首
我试图创建一个 MSI,其中包含 和 exe。在 WIX 中使用了捆绑选项。这样做时出错。有人可以帮我解决这个问题。下面是代码: 错误 error LGH
在 Yii 中,Create 和 Update 通常使用相同的形式。因此,如果我在创建期间有电子邮件、密码、...other_fields...等字段,但我不想在更新期间专门显示电子邮件和密码字段,但
上周我一直在努力创建一个给定一行和一列的 QModelIndex。 或者,我会满足于在已经存在的 QModelIndex 中更改 row() 的值。 任何帮助,将不胜感激。 编辑: QModelInd
出于某种原因,这不起作用: const char * str_reset_command = "\r\nReset"; const char * str_config_command = "\r\nC
现在,我有以下由 original.df %.% group_by(Category) %.% tally() %.% arrange(desc(n)) 创建的 data.frame。 DF 5),
在今天之前,我使用/etc/vim/vimrc来配置我的vim设置。今天,我想到了创建.vimrc文件。所以,我用 touch .vimrc cat /etc/vim/vimrc > .vimrc 所
我可以创建一个 MKAnnotation,还是只读的?我有坐标,但我发现使用 setCooperative 手动创建 MKAnnotation 并不容易。 想法? 最佳答案 MKAnnotation
在以下代码中,第一个日志语句按预期显示小数,但第二个日志语句记录 NULL。我做错了什么? NSDictionary *entry = [[NSDictionary alloc] initWithOb
我正在使用与此类似的代码动态添加到数组; $arrayF[$f+1][$y][$x+1] = $value+1; 但是我在错误报告中收到了这个: undefined offset :1 问题:尝试创
我是一名优秀的程序员,十分优秀!