繁体   English   中英

访问线程环境块时进入程序恐慌状态

[英]Go program panics when accessing the Thread Environment Block

我正在使用Go库进行开发,以访问一些内部Windows线程结构(线程环境块),这需要编写一些汇编代码。 我一直在试图理解为什么它可以在Win32 C ++应用程序上工作,但是在我的Go库上却不能。

Go汇编代码的此代码段访问fs:[0x18]以返回指向该线程的关联TEB的指针:

    // func ReadFsDword(offset uint32) (dword uint32)
    TEXT ·ReadFsDword(SB),$0-8
            MOVL offset+0(FP), AX
            // mov eax, dword ptr fs:[eax]
            BYTE $0x64; BYTE $0x8B; BYTE $0x00
            MOVL AX, ret+8(FP)
            RET

这是等效的MASM代码,可以在MSVC上编译并正常运行:

void* readfsdword(unsigned offset_)
{
    unsigned dw;

    __asm {
        mov eax, offset_
        mov eax, fs:[eax]
        mov dw, eax
    }

    return (void*)dw;
}

访问返回的指向TEB的指针时,Go程序会出现紧急情况。 这是我收到的消息:

紧急情况:运行时错误:无效的内存地址或nil指针取消引用[信号0xc0000005代码= 0x0地址= 0x0 pc = 0x498d5b]

Go汇编代码对我来说似乎是正确的,但是我不明白该程序如何以及为什么会慌乱。 任何帮助深表感谢!

这是重现此问题的示例:

intrinsics.s

#include "textflag.h"
#include "funcdata.h"

// func ReadFsDword(offset uint32) (ret uint32)
TEXT ·ReadFsDword(SB),$0-8
        MOVL offset+0(FP), AX
        // mov eax, dword ptr fs:[eax]
        BYTE $0x64; BYTE $0x8B; BYTE $0x00
        MOVL AX, ret+8(FP)
        RET

intrinsics.go

package nt

func ReadFsDword(offset uint32) (ret uint32)

test.go

package main

import "nt"

func main() {
    GetProcAddress("LoadLibraryExW")
}

func GetProcAddress(proc string) unsafe.Pointer {
    teb := nt.NtGetTeb()
    fmt.Printf("%p", teb)

    // todo: implement
    return nil
}

该问题已解决。 显然,Windows在x64上使用偏移量为0x30的gs寄存器,而在x86 / WoW64模式下使用fs和偏移量为0x18的gs寄存器。 解决方案是根据GOARCH的值使用fsgs以及各自的偏移量。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM