你能拼凑出以下线索来帮助我读取温邦W83793芯片的温度吗?

本文关键字:W83793 读取 芯片 温度 帮助 线索 | 更新日期: 2023-09-27 18:06:16

我正试图让OpenHardwareMonitor从我的超微X7DWA主板上的Winbond W83793芯片读取温度数据。问题是我没有任何低级别的编程经验,而且网上可用的文档似乎不足以解释如何访问温度。

然而,在我研究这个问题的一个月里,我发现了一些值和低级方法,它们可能是解决我的问题的关键。我只需要想出如何利用它们来得到我想要的。这就是我向你求助的地方,因为你可能明白这些信息的含义,以及如何应用它,不像我。我已经试过了,结果出现了很多蓝屏和电脑重启。猜够了,我得把这些线索拼凑起来。以下是我目前所知道的:

  1. 要从芯片中读取,我将以某种方式需要访问SMBus,因为这是监控程序(如CPUID HWMonitor)获取信息的方式。据我所知,OpenHardwareMonitor中没有任何代码来访问SMBus,这就是为什么它可能不会从芯片读取。然而,OpenHardwareMonitor在其Ring0类中包含了以下方法,它使用这些方法从其他芯片访问信息。我可以利用这些方法:

    void Ring0.WriteIOPort(uint port, byte value);
    byte Ring0.ReadIOPort(uint port);
    
  2. 在其他信息中,当我保存报告时,HWMonitor会向我报告以下关于Winbond W83793芯片的信息:

    寄存器空间:SMBus,基址= 0x01100

    SMBus请求:通道0x0,地址0x2F

    看起来这些都是重要的值,但我不知道它们的确切含义,也不知道如何将它们与上面的Ring0方法结合使用。嗯…这么多线索。HWMonitor显示的其他值是实际电压,温度和风扇速度,以及表示芯片上某处数据的十六进制值数组,如果您想查看它,我将在这里重现。

  3. 最后,在W83793数据表的第53页(如果你打开了文档),这里是十六进制的温度地址,我想读(我相信):

    TD1读数-银行0地址1C

    TD2读数- Bank 0地址1D

    TD3读数-银行0地址1E

    TD4读数- Bank 0地址1F

    低位读出-银行0地址22

    TR1读数-银行0地址20

    TR2读数-银行0地址21

到目前为止我只知道这些。OpenHardwareMonitor, W83793芯片和Ring0代码可通过上面提供的链接获得。就像我之前说的,我已经研究了一个月了,但我还是没能解开这个谜团。我希望你能帮助我。所有这些信息可能看起来有点吓人,但我相信它对具有低级编程经验的人来说是有意义的。

总结一下我的问题,使用上面提供的线索来弄清楚如何让OpenHardwareMonitor读取W83793芯片的温度。我不需要在OpenHardwareMonitor中创建芯片的详细信息。我已经准备好了一节课。我只需要编写Ring0命令的顺序和格式,如果这是我需要做的。

编辑:我发现了一些更多的信息。我从HWMonitor打印了一个SMBus设备报告,除其他事项外,我得到了这一行,包括在这里,因为它说0x2F:

SMB设备:I/O = 0x1100,地址0x2F,通道= 0

这一定意味着我需要以某种方式将I/O的地址与芯片的地址结合起来,这似乎是0x2F。我试着把它们加在一起,但是我得到的所有温度读数都是255,所以这不是正确的猜测。

你能拼凑出以下线索来帮助我读取温邦W83793芯片的温度吗?

IO方法就是您所需要的。在x86硬件上,实际上有两个地址池,而不是一个。一个用于存储,当读取指令时被芯片引用,并且有数千种有用和方便的访问方法。另一种是用于寻址外部芯片,具有非常有限和相对较慢的读写操作集。您已经确定的方法使您可以访问第二个区域。

由于您想要读取的寄存器位于银行0中,因此首先需要在芯片上选择银行0,如第12页所示。根据8.1.2.1节中的图表,您需要将0x80写入地址00。根据你的报告,芯片的基址是0x01100,这应该意味着通过WriteIOPort将0x80写入0x01100。

那么你应该能够通过ReadIOPort从0x01100+0x1c, 0x01100+0x1d等读取你想要的值。

我没有时间完全消化你链接到的文档,但这些都是合理的猜测。一些芯片有一个稍微复杂的过程,你必须写一个值,然后确认结果,但我没有看到任何类似的文档。您还需要警惕多任务处理—如果您的代码在设置bank 0和读取相关寄存器之间中断,那么中间的某些进程可能会设置其他bank,从而导致您读取的值是任意其他值。我认为OpenHardwareMonitor有某种机制来处理这个问题,但如果您尝试快速纯用户空间实现并偶尔得到奇怪的结果,则值得记住。

最后,OpenHardwareMonitor的作者好心地帮助了我,现在我可以从芯片上读取温度了。虽然这个问题的整个解决方案有点复杂,而且仍然超出了我的能力范围,但这里是使用Ring0类的基本读写,可供任何感兴趣的人使用。请注意,这是特定于我的机器和芯片。对你来说,基址和从址可能不同,但是你可以使用CPUID HWMonitor,通过打印一个报告来找到它们。

首先,这里是使用的常量:

private const int   BASE_ADDRESS    = 0x1100;
private const uint  SLAVE_ADDRESS   = 0X2F; // as we figured out already
private const byte  HOST_STAT_REG   = 0;    // host status register
private const byte  HOST_BUSY       = 1;    
private const byte  HOST_CTRL_REG   = 2;    // host control register
private const byte  HOST_CMD_REG    = 3;    // host command register
private const byte  T_SLAVE_ADR_REG = 4;    // transmit slave address register
private const byte  HOST_DATA_0_REG = 5;  
private const byte  BYTE_DATA_COMM  = 0x08; // byte data command
private const byte  START_COMM      = 0x40; // start command
private const byte  READ            = 1;
private const byte  WRITE           = 0;

接下来,下面是从芯片上的寄存器中读取特定字节的基本代码:

// first wait until ready
byte status;
do
{
  status = Ring0.ReadIoPort(BASE_ADDRESS + HOST_STAT_REG);
} while ((status & HOST_BUSY) > 0);
if ((status & 0x1E) != 0)
{
  Ring0.WriteIoPort(BASE_ADDRESS + HOST_STAT_REG, status);
}
// now get the value
Ring0.WriteIoPort(BASE_ADDRESS + HOST_DATA_0_REG, 0);
Ring0.WriteIoPort(BASE_ADDRESS + HOST_COMM_REG, theRegister)
Ring0.WriteIoPort(BASE_ADDRESS + T_SLAVE_ADR_REG, 
            (byte)((SLAVE_ADDRESS << 1) | READ));
Ring0.WriteIoPort(BASE_ADDRESS + HOST_CTRL_REG,
            START_COMM | BYTE_DATA_COMM);
Ring0.ReadIoPort(BASE_ADDRESS + HOST_DATA_0_REGISTER); // this returns the value
// now wait for it to end
while ((Ring0.ReadIoPort(BASE_ADDRESS + HOST_STAT_REG) & HOST_BUSY) > 0) {}

虽然我不是很了解它,但这可以作为一个粗略的指导,比我经验更低的人遇到类似的问题。