使用 MIPS 汇编语言理解递归
我在课堂上,我们已经/正在讨论汇编语言中的递归。 我觉得我理解了递归,但是越多的人试图向我解释它,我就越觉得离它很遥远。
无论如何,我们的最后一张幻灯片有一个函数(用 C 语言?),老师说他会在课堂上介绍它,但要求我们学生在黑板上展示课堂上的其余部分。我感觉他一直在看着我,我害怕看起来很愚蠢。
你们能帮我在 MIPS 中编写这段代码并帮助我理解它吗? 如果这太难了,我不知道
用 MIPS 汇编语言编写来查找修复(i,x),其中修复(i,x)是 递归地定义为:
int fix(int i, int x) // assume i > 0, x > 0
{
if (x>0)
return fix(i,x-1);
else if (i>0)
return fix(i-1, i-1)+1;
else
return 0;
}
谢谢你们,我的课是明天,我还是希望他永远不要打电话给我;但我想真正理解这些材料。
注意:这将是课堂练习,附有 0 学分。我觉得班上的每个人都已经知道如何做到这一点。
I was in class and we have/are covering recursion in Assembly Language.
I felt like I understood recursion, but the more people attempt to explain it to me, the more I feel distant from it.
Anyways, our last slide had a function (in C?) and the teacher said he will cover it in class but call on us students to show the rest of the class on the board. I felt like he was looking at me the whole time and I am terrified of looking stupid.
Can you guys help me write this code in MIPS and help me understand it?
IDK if this is too hard
Write in MIPS Assembly Language that to find fix(i,x), where fix(i, x) is
defined recursively as:
int fix(int i, int x) // assume i > 0, x > 0
{
if (x>0)
return fix(i,x-1);
else if (i>0)
return fix(i-1, i-1)+1;
else
return 0;
}
Thank you guys, my class is tomorrow and I still hope he never calls on me; but I would like to actually understand this material.
NOTE: This will be an in class exercise with 0 credit attached. I feel like everyone in the class already knows how to do this.
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(3)
汇编语言与递归无关,它只是由于 C 语言以及调用约定和实现而起作用。只需在汇编程序中实现 C,而不关心递归。我想我在一个宠物项目 http://github.com/dwelch67/lsasim 上谈到了这一点,除非我改变它最后一课是手动将 C 中的递归转换为汇编程序。它不是 mips,所以不用担心这是一个家庭作业问题。
无论如何,开始的关键是简单地在汇编中实现 C。
例如,您有一个带有输入参数的函数。
您需要声明自己的调用约定或使用现有的调用约定,要实现 C,这意味着您需要一些位置来存放输入参数,要么将它们压入堆栈,要么将它们带入寄存器。假设没有优化,您可以在整个函数中保留这些变量并在最后进行清理。因此,如果您需要在代码中的任何位置调用 ANY 函数(递归,调用相同的函数,是 ANY 的一个非常小的子集,但属于该类别并且不是特殊的),您需要保留这些变量。如果调用约定将它们带入堆栈,那么您就已经完成了,如果调用约定将它们带入寄存器,那么您需要在调用之前保留它们,并在调用之后恢复
并继续实现该函数。
就是这样,剩下的就自然而然了。
如果您碰巧注意到您作为示例创建的函数,在该函数内调用函数后没有需要保留输入变量的路径。并且输入变量被修改并用作下一次调用的输入。因此,对 C 代码实现的优化是不用担心保留这些变量。只需修改它们并传递它们即可。在调用约定中使用寄存器将是对此特定函数执行此操作的最简单方法。这是编译器在优化时无论如何都会做的事情(如果使用基于寄存器的调用约定则不保留)。
如果这就是所谓的,您也可以进行尾部优化。通常,当调用函数时,您可以使用指令通常执行的任何操作来执行“调用”,这与简单的跳转或分支不同,因为有一个返回值保存在某处。并且有某种返回函数可以撤消此操作并在调用后返回到指令。嵌套调用意味着嵌套返回值,跟踪所有返回值。在这种情况下,以及其他情况下,您在函数的执行路径中所做的最后一件事是调用另一个函数,您可以改为(取决于指令集)分支到该函数,而不必嵌套另一组返回值。以arm指令集为例:
某些代码调用第一个函数:
bl firstfun:
在arm中,bl表示分支链接。寄存器 14 将填充返回值,程序计数器将填充函数firstfun 的地址。
通常,如果您需要从函数调用函数,则需要保存 r14,以便可以从该函数返回,而无需尾部优化:
bx lr 仅意味着分支到 r14 中的内容,在本例中是返回。优化看起来像这样,重要的是要注意,在第一个函数中,对第二个函数的调用是从第一个函数返回之前所做的最后一件事。这是本次优化的关键。
b 只是表示分支,而不是分支链接,它只是修改 pc 而不会修改 r14 或任何其他寄存器或堆栈。两个实现的执行在功能上是相同的,外部函数调用firstfun,并且在正确的执行路径中有一个返回(bx r14)。
其他人指出,由于您将原始调用者返回零,因此该代码可能会完全优化自身而变得无用。
The assembly language has nothing to do with the recursion, that just happens to work due to the C language and the calling conventions and implementation. Just implement the C in assembler and dont care about recursion. I think I touched on this on a pet project http://github.com/dwelch67/lsasim unless I changed it out the last lesson is manually converting recursion in C to assembler. Its not mips so no worries about this being a homework problem.
Anyway the key to starting is to simply implement the C in assembly.
For example you have a function with input parameters.
You will need to declare yourself a calling convention or use an existing one, to implement C this means you need some place for the input parameters, either push them on the stack or bring them in in registers. Assuming no optimization, you preserve these variables throughout the function and clean up at the end. So if you need to call ANY function to anywhere in the code (recursion, calling the same function, is a very small subset of ANY but falls into that category and IS NOT SPECIAL) you need to preserve these variables. IF the calling convention brings these in on the stack then you are already done, if the calling convention brings these in in registers then you need to preserve them before the call and restore after
and continue implementing the function.
that is it, the rest will take care of itself.
If you happen to notice that the function you created as an example, does not have a path where the input variables need to be preserved after the call to a function within this function. And the input variables are modified and used as inputs to the next call. so an optimization to your implementation of the C code would be to not worry about preserving those variables. simply modify them and pass them on. Using registers in the calling convention would be the simplest way to do this for this specific function. This is what a compiler would do anyway when optimizing (not preserve if using register based calling convention).
You could also do a tail optimization if that is what it is called. Normally when calling a function you use whatever the instruction normally does to perform a "call" which is different from a simple jump or branch because there is a return value kept somewhere. And there is some sort of return function that undoes this and returns back to the instruction after the call. nested calls mean nesting the return values, keeping track of all of them. IN this case though and other cases where the last thing you do the execution path of a function is call another function, you can instead (depends on the instruction set) branch to the function and not have to nest another set of return values. Looking at the arm instruction set for example:
Some code calls the first function:
bl firstfun:
In arm bl means branch link. register 14 will be filled in with the return value and the program counter will be filled in with the address to the function, firstfun.
typically if you need to call a function from a function you need to save r14 so you can return from that function, without this tail optimization:
bx lr just means branch to the contents in r14 which in this case is the return. the optimization looks like this, it is important to note that in the first function the call to the second function is the last thing you do before returning from the first function. that is the key to this optimization.
b just means branch, not a branch link simply modifies the pc and does not modify r14 or any other register or stack. The execution of the two implementations is the same functionally the outer function makes a call to firstfun and there is a return (bx r14) in the proper execution path.
Other folks have pointed out that this code may completely optimize itself into nothing since you return zero the original caller.
就像上面的C一样把它分成3块。通过寄存器传递值 i 和 x 并在运行 if 检查并修改寄存器后调用自己。上述内容不应花费超过 30 行汇编程序来完成。如果我是一名更好的 MIPS 编码员,我会为你做这件事。它看起来像(是伪汇编程序),
有趣的是,这将始终返回 0,如 C 中编写的:)
Just break it into 3 pieces like the C above. Pass the values i and x via register and call yourself after running the if check and modifying the registers. The above shouldn't take more than 30ish lines of assembler to do. If I were a better MIPS coder, I would do it for you. It would look something like (is psuedo assembler)
Funny thing is, this will always return 0 as written in C :)
虽然我反对只给出家庭作业问题的答案,但这里有一个等效的函数,可以找到
fix(i, x)
,并带有示例调用(这个版本比 C 版本更高效) ):这对您来说是一个教训,让您在尝试编写函数之前了解函数的用途:)
While I am against just giving the answer to homework problems, here is the equivalent function that will find
fix(i, x)
, complete with example call (this version is a little more efficient than the C version):And let this be a lesson to you to learn what functions do before attempting to code them :)