我想写入PMC1寄存器,所以我想将RCX设置为188.
我用来使用WRMSR指令的代码最后附上。问题是我将eax和ecx值(64位)传递给宏,但%rcx寄存器始终设置为eax值。
#define RTXEN_WRITE_MSR(eax, ecx) __asm__ __volatile__(\
"movq %0, %%rax\n\t"\
"xorq %%rdx, %%rdx\n\t"\
"xorq %%rcx, %%rcx\n\t"\
"movq %1, %%rcx\n\t"\
"wrmsr"\
:\
:"r" (eax), "r" (ecx)\
:\
)
uint64_t eax = 0x14f2e
uint64_t edx = 0x188
printk("eax:%#018lx, edx:%#018lx\n", eax, edx);
RTXEN_WRITE_MSR(eax, ecx);
执行RTXEN_WRITE_MSR(eax,ecx)时,内核出现紧急情况! 注册信息如下:
(XEN) CPU: 1
(XEN) RIP: e008:[<ffff82c4c02166b1>] setread_perf_counter+0x251/0x4c0
(XEN) RFLAGS: 0000000000010046 CONTEXT: hypervisor
(XEN) rax: 0000000000014f2e rbx: ffff8308558f7718 **rcx: 0000000000014f2e**
(XEN) rdx: 0000000000000000 rsi: 000000000000000a rdi: ffff82c4c0270640
(XEN) rbp: 0000000000000001 rsp: ffff83086bec7cd8 r8: 0000000000000004
(XEN) r9: 0000000000000004 r10: 0000000000000004 r11: 0000000000000400
(XEN) r12: 0000000000014f2e r13: 0000000200000000 r14: ffff83086bec7dd8
(XEN) r15: 00000000000000fb cr0: 000000008005003b cr4: 00000000001426f0
(XEN) cr3: 000000008ce6a000 cr2: ffff880115ca32a0
(XEN) ds: 002b es: 002b fs: 0000 gs: 0000 ss: e010 cs: e008
任何人都可以帮我看看我的代码中有什么问题吗?
答案 0 :(得分:1)
你知道你可以让你的编译器立即在正确的寄存器中传递你的参数吗?
asm volatile (
"wrmsr"
:
: "a" ((uint32_t)eax)
, "c" ((uint32_t)ecx)
, "d" ((uint32_t)0)
);
uint32_t
的类型转换确保编译器将使用32位寄存器变体(eax
,ecx
,edx
)。
您可以在此处找到英特尔寄存器的约束修饰符:http://gcc.gnu.org/onlinedocs/gcc/Machine-Constraints.html#Machine-Constraints
旁注:如果您在asm中使用寄存器,那么您需要告诉编译器您正在这样做。您需要将您使用的所有寄存器和编译器在第三个冒号后面的clobber寄存器列表中不知道。