QEMU arm 大小端问题

在 qemu2.7.1 上开发 tms570ls3137 板级设备时,遇到了一个大小端的问题。本文将详细描述该问题,并对测试过程进行记录。

问题描述

  • 目前 PC 上以 x86 体系结构为代表的小端机为主,因此在 qemu 最初的实现里只考虑到了对小端机的支持。但是目前仍有一部分以 arm 为核心的板子采用大端序。
  • 因此在 2016 年,qemu 增加了对 arm 大端的支持,详细的内容可以查看qemu 开发邮件
  • 即便如此,我在开发时依旧遇到了一些问题,包括:gdb 调试地址大小端出错,数据顺序颠倒等。

qemu 中 arm 大小端解决方式

在真实硬件中 arm 芯片是可以通过设置 sctlr 和 cpsr 寄存器切换大小端模式的,详细内容可以查看芯片手册。对一块板子来说大小端限制一般来自于外设。

QEMU 启动虚拟板级设备时首先会读取 elf 文件的 header 从而判断该文件是大端或是小端,在得知 elf 文件的大小端信息后,QEMU 会去设置 arm cpu 的 sctlr 和 cpsr,从而切换到想要的模式并设置info->endiannes,这样就使 QEMU 上的 arm cpu 即可运行小端程序也可运行大端程序。如果想要在自己实现的板子上限制大小端,则可设置外设中的MemoryRegionOps.endianness属性。

关键代码/qemu2.7.1/hw/arm/boot.c

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
static void do_cpu_reset(void *opaque)
{
    ARMCPU *cpu = opaque;
    CPUState *cs = CPU(cpu);
    CPUARMState *env = &cpu->env;
    const struct arm_boot_info *info = env->boot_info;

    cpu_reset(cs);
    if (info) {
        if (!info->is_linux) {
            int i;
            /* Jump to the entry point.  */
            uint64_t entry = info->entry;

            switch (info->endianness) {
            case ARM_ENDIANNESS_LE:
                env->cp15.sctlr_el[1] &= ~SCTLR_E0E;
                for (i = 1; i < 4; ++i) {
                    env->cp15.sctlr_el[i] &= ~SCTLR_EE;
                }
                env->uncached_cpsr &= ~CPSR_E;
                break;
            case ARM_ENDIANNESS_BE8:
                env->cp15.sctlr_el[1] |= SCTLR_E0E;
                for (i = 1; i < 4; ++i) {
                    env->cp15.sctlr_el[i] |= SCTLR_EE;
                }
                env->uncached_cpsr |= CPSR_E;
                break;
            case ARM_ENDIANNESS_BE32:
                env->cp15.sctlr_el[1] |= SCTLR_B;
                break;
            case ARM_ENDIANNESS_UNKNOWN:
                break; /* Board's decision */
            default:
                g_assert_not_reached();
            }

            if (!env->aarch64) {
                env->thumb = info->entry & 1;
                entry &= 0xfffffffe;
            }
            cpu_set_pc(cs, entry);
        } else {
           ……
    }
}

外设(如:pl011)字节序设置/qemu2.7.1/hw/char/pl011.c

1
2
3
4
5
static const MemoryRegionOps pl011_ops = {
    .read = pl011_read,
    .write = pl011_write,
    .endianness = DEVICE_NATIVE_ENDIAN,//修改为 DEVICE_BIG_ENDIAN 或 DEVICE_LITTLE_ENDIAN
};

解决方法

在我寻找解决方法的时候,发现已经有人遇到过相同的问题,并且在 Qemu-devel 中有相关的邮件记录

其中提到的情况与我遇到的完全相符:

  • 通过 gdb 调试 qemu,qemu 中 pc 值完全正确,并且进入到了外设读写数据的相关流程;
  • gdb 调试裸金属程序,gdb 中的地址以错误的大小端模式显示;
  • 使用 x 命令打印时大小端正常;
  • 数据的顺序出错,如:HelloWorld 变成了 lleHlroW;

根据邮件列表中的 patch 将代码加上,就可以了。

注意:在添加完 patch 之后,还需要修改一个地方,否则在发生中断之后大小端依然会出错。

qemu/target-arm/helper.c文件中

1
2
3
4
5
6
7
8
9
@ -6418,7 +6418,7 @@ static void arm_cpu_do_interrupt_aarch32(CPUState *cs)
    /* Set new mode endianness */
    env->uncached_cpsr &= ~CPSR_E;
    if (env->cp15.sctlr_el[arm_current_el(env)] & SCTLR_EE) {
-       env->uncached_cpsr |= ~CPSR_E; //set new mode endianness
+       env->uncached_cpsr |= CPSR_E;
    }
    env->daif |= mask;
    /* this is a lie, as the was no c1_sys on V4T/V5, but who cares

这个问题在邮件中也有提到,但是 patch 中并没有修改,因此需要格外注意。

tms570 大端 be8 与 be32 之谜

tms570 是大端机,但是它属于大端的那个模式呢,在 tms570 手册中,它可以使用 be32 模式,但是在 cortexr4f 的文档中,我发现,它是不能使用 be32 模式,只能使用 be8 模式。

cortexr4f 手册

同时,当我在网上查询这个问题的时候,发现早也有人对此提出疑问,在 QMEU开发邮件中可看到详细信息。

同时,长期使用在 tms570 上开发的朋友却告诉我,在真实硬件上是使用 be32 模式。

因此这里应该是手册描述错误。

这里不是错误。。。。。这是由于 TI 对 be32 做了转换,把它转换成了 be8

官方答复

tms570 实现 be32 转 be8

在第 4 节里,发现了 tms570 大端模式中描述冲突的问题,官方老哥回复的比较简单

We don't modify the Cortex-R4 core, but add a bridge to convert BE32 to BE8.

没办法,只能自己实现 be32 到 be8 的转换了。

实现这个功能的思路有两个:

  1. 将 be32 文件转换为 be8;
  2. 基于已有的 le、be32 和 be8 模型,在 qemu 内部添加一个针对 tms570 的内存读写控制;

目前我根据思路 2,实现了该功能。