0

在我的机器语言课程中,我们必须将 c++ 翻译成 mips,问题是我不太了解 c++,只是基本的东西,所以我很难理解这个函数:

void IPtoDD(int arg0, char *arg1)
{
  int temp, numChar, shift = 24;

  for (int i=0; i<4; i++) {
    temp = arg0 >> shift;
    temp = temp & 0x000000ff;
    numChar = byteToDec(temp,arg1);
    arg1 += numChar;
    *arg1++ = '.';
    shift -= 8;
  }

尤其是

    temp = arg0 >> shift;
    temp = temp & 0x000000ff;
4

4 回答 4

2

>>运算符是“右移运算符”。它将左侧值的二进制表示向右移动 x 位,其中 x 是右侧的数字。这看起来像什么?这里有些例子:

decimal: 128                   >> 1 == 64
binary:  0b0000 0000 1000 0000 >> 1 == 0b0000 0000 0100 0000

decimal: 64                    >> 1 == 32
binary:  0b0000 0000 0100 0000 >> 1 == 0b0000 0000 0010 0000

请注意大二进制数中的 1 如何向右移动 1。在您的代码中,您移动arg0的是shift,其中arg0是一些输入并且shift是 24、16、8,然后是 0,因为您正在循环。这意味着第一个循环,您arg0向下移动 24 位,然后将其存储到 temp.txt 中。

第二个语句是按位与,它将左侧的每个位与右侧的每个相应位进行比较。如果两个位都为 1,则相同位置的结果位为 1,否则为 0。以下是一些示例:

decimal: 7      & 15     == 7
binary:  0b0111 & 0b1111 == 0b0111

decimal: 5      & 12     == 4
binary:  0b0101 & 0b1100 == 0b0100

在您的表达式中,AND 使用右手边的值完成,该值0x000000FF是 255 的十六进制等效值,以十进制表示。

您感兴趣的两个陈述正在做的是从输入中提取每个“字节”或“8位块” arg0

temp = arg0 >> shift;
temp = temp & 0x000000ff;

Input:  arg0 = 0x12345678, shift = 24
Output: temp = 0x12

Input:  arg0 = 0x12345678, shift = 0
Output: temp = 0x78

关于右移的附录:

您正在这里移动一个有符号整数,但幸运的是,您使用以下 AND 掩码屏蔽了这些位。为了完整起见,对有符号整数右移可以做两件事之一(假设这里是 16 位数字,32 位或 64 位数字的结果会有所不同):

算术右移:

decimal: -128                  >> 1 == -64
binary:  0b1111 1111 1000 0000 >> 1 == 0b1111 1111 1100 0000

逻辑右移:

decimal: -128                  >> 1 == 32704
binary:  0b1111 1111 1000 0000 >> 1 == 0b0111 1111 1100 0000

请注意算术右移如何“复制”最高位,其中“逻辑右移”引入了零。它是如何实际解释的特定于平台的,所以如果可能的话尽量远离它,并坚持移动无符号整数。

于 2013-10-10T19:23:04.827 回答
0
temp = arg0 >> shift

这会对 的内容进行二进制右移arg0。它执行此移位shift次数。因此,由于shift = 24,它将arg0向右移动 24 次。

temp = temp & 0x000000ff;

这掩盖了temp. 最后一个字节仍然存在,因为任何&'ed 都只f反映原始值。

于 2013-10-10T19:15:22.430 回答
0
temp = arg0 >> shift

这是一个位转移。整数值arg0将向右移动shift位数。

例如,如果 shift = 8 且 arg0 = 0x1234,则 temp 将分配为 0x12。

在您的代码中,如果 arg0 = 0x12345678 且 shift = 24,则 temp 变为 0x123456。

temp = temp & 0x000000ff;

这是按位and运算。十六进制数 0x0000000ff 用于在传递到 byteToDec() 之前屏蔽最低有效字节。

因此,在最后一个示例中,arg0 = 0x12345678,temp 移动 24 到 = 0x123456,然后掩码为 0x00000056。

就 MIPS 而言, -- 右移逻辑变量 (SRLV) 指令将用于移位

SRLV $d, $t, $s

其中 $d 是目标寄存器,$t 是源寄存器,$s 是保存移位次数值的寄存器。值得理解的是,这种移位会从左侧引入零,并将更改有符号整数的值为负数,但由于应用了一个掩码,该掩码将零应用于除最低有效字节之外的所有字节,它可以为您节省一些指令。

于 2013-10-10T19:30:32.947 回答
0

在这一行

temp = arg0 >> shift;

arg0 的值从最高有效位移动到最低位(注意它可能是从左到右或从右到左取决于 CPU 的字节序)移位值(第一次 24 然后 16 然后 8)

temp = temp & 0x000000ff;

二进制 AND 应用于 temp 即除次要 8 位之外的所有重置为 0,次要 8 位不更改。如果您需要在汇编中实现它,您可能不需要执行二进制 AND,而是获取正确的字节。

于 2013-10-10T19:20:40.437 回答