HP 2114/15/16浮点转换 [英] HP 2114/15/16 floating point conversion

查看:79
本文介绍了HP 2114/15/16浮点转换的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

我有一些文件中带有浮点值,因此在确定其确切编码时遇到了问题。我尝试了几种方法将其转换为标准的double值,但是运气不高。我知道这些值应该转换为什么,但是需要一种直接从文件中提取的方法。

I have some files with floating point values in them that I'm having problems figuring out their exact encoding. I've tried several ways to convert to a standard double value, but haven't had much luck. I know what the values are supposed to convert to, but need a method to extract directly from the files.

这些是HP-1​​000(HP21xx)系列浮点值。类似于48位Pascal,但不相同。

These are HP-1000 (HP21xx) series floating point values. Similar to 48 bit Pascal, but not the same.

除了一些旧的,不可靠的文档(无法使用其声称的格式进行转换),并且从quadibloc.com获得的格式列表我什么都没找到。 quadibloc的格式如下:

Other than some old, unreliable documentation (couldn't get a conversion using what it claimed was the format), and a list of formats from quadibloc.com I have found nothing else. Format from quadibloc gives the following:

(Note, big-endian order)

MSb: mantissa sign (says it's 2s complement)

39 bits: mantissa

7 bits: exponent

1 bit: exponent sign

其他文档尚不清楚,但似乎是全2补码。
我已经尝试过找到的Pascal转换代码,但是还差得远。 (将指数符号移动到字节的高位之后。)

The other documentation isn't clear, but seems to say it's all 2s complement. I've tried the Pascal conversion code I found, but it doesn't even come close. (after moving the exponent sign to high bit of the byte).

示例,使用模拟HP-1000的PPC进行了转换:

Examples, converted using a PPC that emulates an HP-1000:

A7EB851EB90A    -22.02

A870A3D70A0A    -21.89

A8AE147AE20A    -21.83

A8E147AE140A    -21.78

A9666666670A    -21.65

AC70A3D70A0A    -20.89

ACC28F5C290A    -20.81

ACCCCCCCCC0A    -20.8

AE70A3D70B0A    -20.39

AEB851EB850A    -20.32

这是从只有一个文件。我至少要提取其中100个文件。

This was from only one file. I have at least 100 of these files to extract from.

有什么想法吗?肯定会感激的。

Any ideas? Sure would be appreciated.

编辑:我想我应该说过我使用的语言。在这种情况下,它是C#。至于其他数据,我有很多。这里有更多示例,包括负指数,至少当转换为小数时。

I guess I should have said what language I am working in. In this case, it's C#. As for additional data, I have a ton of it. More examples here, these include negative exponents, at least when converted to decimal.

400000000002    1
43851EB85204    2.11
451EB851EB04    2.16
4C7AE147AE04    2.39
4EC4EC4EC5F3    4.8076923077E-03
519999999A04    2.55
5838B6BE9BF3    5.3846153846E-03
5B851EB85202    1.43
5BD70A3D7108    11.48
5C7AE147AE08    11.56
5E51EB851E08    11.79
5FD70A3D7108    11.98
62E147AE1508    12.36
64A3D70A3E08    12.58
666666666702    1.6
67AE147AE202    1.62
6B204B9E54F3    6.5384615385E-03
733333333302    1.8
762762762AF3    7.2115384616E-03
794DFB4619F3    7.4038461538E-03
7C74941627F3    7.5961538465E-03
7E07E07E14F3    7.6923076925E-03

以前应该包含正数。

尝试了建议,但使用C#。结果截然不同。我什至拆分了计算的位数,但与建议中给出的数字相比,我得到了很高的数字。这是扩展了位的代码。

Tried the suggestions, but in C#. Wildly different results. I even split out the bits of the calculations, but I get very high numbers compared to the numbers given in the suggestions. Here's the code with the bits expanded.

    Int64[] test_data = new Int64[] {
        0xA7EB851EB90A, 0xA870A3D70A0A, 0xA8AE147AE20A, 0xA8E147AE140A,
        0xA9666666670A, 0xAC70A3D70A0A, 0xACC28F5C290A, 0xACCCCCCCCC0A,
        0xAE70A3D70B0A, 0xAEB851EB850A, 0x400000000002, 0x43851EB85204,
        0x451EB851EB04, 0x4C7AE147AE04, 0x4EC4EC4EC5F3, 0x519999999A04,
        0x5838B6BE9BF3, 0x5B851EB85202, 0x5BD70A3D7108, 0x5C7AE147AE08,
        0x5E51EB851E08, 0x5FD70A3D7108, 0x62E147AE1508, 0x64A3D70A3E08,
        0x666666666702, 0x67AE147AE202, 0x6B204B9E54F3, 0x733333333302,
        0x762762762AF3, 0x794DFB4619F3, 0x7C74941627F3, 0x7E07E07E14F3
    };

    private void Checkit()
    {
        Byte[] td = new Byte[6];                        // 6 byte array for reversed data
        Byte[] ld = new Byte[8];                        // 8 byte conversion array

        for (Int32 i = 0; i < test_data.Length; i++)    // Loop through data
        {
            ld = BitConverter.GetBytes(test_data[i]);   // Get value as byte array
            for (Int32 j = 0; j < 6; j++)               // Copy the bytes in reverse
                td[5 - j] = ld[j];
            // Go test them
            Console.WriteLine("{0:X6}  --> {1:E}", test_data[i], Real48ToDouble(ref td));
        }
    }

    Double Real48ToDouble(ref Byte[] realValue)
    {
        // Values are using input value of A7EB851EB90A and 7E07E07E14F3
        if (realValue[0] == 0)
            return 0.0;                                     // Null exponent = 0

        Byte[] b = new Byte[8] { 0, 0, 0, 0, 0, 0, 0, 0 };  // 64 bit byte array
        for (Int32 i = 0; i < 5; i++)                       // Copy over the 48 bit info
            b[4 - i] = realValue[i];

        Int64 mant = BitConverter.ToInt64(b, 0);            // 0x000000A7EB851EB9 - Get mantissa with sign
                                                            // 0x0000007E07E07E14
        Int32 expo = realValue[5];                          // 0x0000000A - Grab the exponent
                                                            // 0x000000F3
        Int32 mant_sign = (Int32)(mant >> 39);              // 0x00000001
                                                            // 0x00000000

        // sign extend mantissa from 40 to 64 bits, then take absolute value
        mant = (Int64)((mant ^ (1L << 39)) - (1L << 39));   // 0xFFFFFFA7EB851EB9 - First calc
                                                            // 0x0000007E07E07E14
        mant = Math.Abs(mant);                              // 0x00000058147AE147 - Make absolute
                                                            // 0x0000007E07E07E14

        // convert mantissa to floating-point
        Double fmant = mant * Math.Pow(2, -39.0);           // 0.68812499999876309 - Second calc
                                                            // 0.98461538463743636
        // rotate exponent field right by 1 bit
        expo = (expo >> 1) | ((expo & 1) << 7);             // 0x00000005
                                                            // 0x000000F9
        // sign extend exponent from 8 to 32 bits
        expo = ((expo ^ (1 << 7)) - (1 << 7));              // 0x00000005
                                                            // 0xFFFFFFF9

        // compute scale factor from exponent field
        Double scale = Math.Pow(2, expo);                   // 32.0 - Scale
                                                            // 0.0078125

        // scale mantissa, and apply sign bit for final result
        Double num = fmant * scale;                         // 22.019999999960419 - Make the final abs number
                                                            // 0.0076923076924799716

        return (mant_sign != 0) ? (-num) : num;             // -22.019999999960419 - Return with sign
                                                            // 0.0076923076924799716
    }

已更改
以上的代码现在可以正常工作。

Changed code above The code above is now working correctly.

推荐答案

《惠普杂志》,1978年10月,,第7页。图14提供了有关此处使用的浮点格式的以下信息:

An article in Hewlett-Packard Journal, October 1978, pg. 14 gives the following information about the floating-point format used here:


扩展精度数字具有39位带符号的尾数,并且相同
七位带符号的指数。所有尾数均已标准化,这意味着
处在[½,1)和[-1,-½)范围内。

Extended-precision numbers have a 39-bit signed mantissa and the same seven-bit signed exponent. All mantissas are normalized, which means they are in the ranges [½, 1) and [-1, -½).

与问题中的信息一起考虑,这意味着尾数是一个有符号的补码数字,由前五个字节表示,是40位整数,必须除以2 39 才能得到其数字浮点值。

Considered together with the information in the question this means the mantissa is a signed, two's-complement, 40-bit integer represented by the first five bytes, and must be divided by 239 to get its numerical floating-point value.

二进制指数存储在最后一个字节中。根据期刊文章中将40位尾数字段描述为39位尾数的描述,虽然将指数描述为具有7位,以及有关问题中指数符号位的信息,但似乎指数是一个有符号的二进制补码8位整数,已将其向左旋转一位,以便其符号位在指数字节的位[0]中结束。要对其进行处理,我们需要向右旋转一位。

The binary exponent is stored in the last byte. Based on the description of the 40-bit mantissa field as a 39-bit mantissa in the journal article, while the exponent is described a having seven bits, together with information on the exponent sign bit from the question, it appears that the exponent is a signed, two's-complement 8-bit integer, which has been rotated left by one bit so that its sign bit winds up in bit [0] of the exponent byte. To process it, we need to rotate right one bit.

基于上述信息,我编写了以下C99程序,该程序成功地从问题中解码了测试向量:

Based on the above information, I wrote the following C99 program that successfully decodes the test vectors from the question:

#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>
#include <math.h>

int64_t test_data [] =
{
    0xA7EB851EB90A, 0xA870A3D70A0A, 0xA8AE147AE20A, 0xA8E147AE140A,
    0xA9666666670A, 0xAC70A3D70A0A, 0xACC28F5C290A, 0xACCCCCCCCC0A,
    0xAE70A3D70B0A, 0xAEB851EB850A, 0x400000000002, 0x43851EB85204,
    0x451EB851EB04, 0x4C7AE147AE04, 0x4EC4EC4EC5F3, 0x519999999A04,
    0x5838B6BE9BF3, 0x5B851EB85202, 0x5BD70A3D7108, 0x5C7AE147AE08,
    0x5E51EB851E08, 0x5FD70A3D7108, 0x62E147AE1508, 0x64A3D70A3E08,
    0x666666666702, 0x67AE147AE202, 0x6B204B9E54F3, 0x733333333302,
    0x762762762AF3, 0x794DFB4619F3, 0x7C74941627F3, 0x7E07E07E14F3
};
#define NBR_TEST_VECTORS (sizeof(test_data) / sizeof(int64_t))

int main (void)
{
    for (int i = 0; i < NBR_TEST_VECTORS; i++) {

        // extract mantissa and exponent bits
        int64_t mant = test_data[i] >> 8;
        int expo = test_data[i] & 0xFF;
        int mant_sign = mant >> 39;

        // sign extend mantissa from 40 to 64 bits, then take absolute value
        mant = ((mant ^ (1LL << 39)) - (1LL << 39));
        mant = llabs (mant);

        // convert mantissa to floating-point
        double fmant = mant * pow (2.0, -39.0);

        // rotate exponent field right by 1 bit
        expo = (expo >> 1) | ((expo & 1) << 7);

        // sign extend exponent from 8 to 32 bits
        expo = ((expo ^ (1 << 7)) - (1 << 7));

        // compute scale factor from exponent field
        double scale = pow (2.0, expo);

        // scale mantissa, and apply sign bit for final result
        double num = fmant * scale;
        num = mant_sign ? -num : num;

        printf ("%012llx  --> % 20.11e\n", test_data[i], num);
    }
    return EXIT_SUCCESS;
}

以上程序的输出应如下所示:

The output of the above program should look as follows:

a7eb851eb90a  -->  -2.20200000000e+001
a870a3d70a0a  -->  -2.18900000000e+001
a8ae147ae20a  -->  -2.18300000000e+001
a8e147ae140a  -->  -2.17800000000e+001
a9666666670a  -->  -2.16500000000e+001
ac70a3d70a0a  -->  -2.08900000000e+001
acc28f5c290a  -->  -2.08100000000e+001
accccccccc0a  -->  -2.08000000000e+001
ae70a3d70b0a  -->  -2.03900000000e+001
aeb851eb850a  -->  -2.03200000000e+001
400000000002  -->   1.00000000000e+000
43851eb85204  -->   2.11000000000e+000
451eb851eb04  -->   2.16000000000e+000
4c7ae147ae04  -->   2.39000000000e+000
4ec4ec4ec5f3  -->   4.80769230769e-003
519999999a04  -->   2.55000000000e+000
5838b6be9bf3  -->   5.38461538456e-003
5b851eb85202  -->   1.43000000000e+000
5bd70a3d7108  -->   1.14800000000e+001
5c7ae147ae08  -->   1.15600000000e+001
5e51eb851e08  -->   1.17900000000e+001
5fd70a3d7108  -->   1.19800000000e+001
62e147ae1508  -->   1.23600000000e+001
64a3d70a3e08  -->   1.25800000000e+001
666666666702  -->   1.60000000000e+000
67ae147ae202  -->   1.62000000000e+000
6b204b9e54f3  -->   6.53846153847e-003
733333333302  -->   1.80000000000e+000
762762762af3  -->   7.21153846158e-003
794dfb4619f3  -->   7.40384615382e-003
7c74941627f3  -->   7.59615384651e-003
7e07e07e14f3  -->   7.69230769248e-003

这篇关于HP 2114/15/16浮点转换的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆