使用__m256d寄存器(Using __m256d registers)

2019-08-01 10:28发布

如何使用__m256d

说我想要使用英特尔AVX指令_mm256_add_pd一个简单Vector3与3-64位级double精密部件( xy ,和z )。 什么是使用正确的方法是什么?

由于xyz成员 Vector3类,_can我宣布他们在union__m256d变量?

union Vector3
{
  struct { double x,y,z ; } ;
  __m256d _register ;  // the Intel register?
} ;

那我可以走:

Vector3 add( const Vector3& o )
{
  Vector3 result;
  result._register = _mm256_add_pd( _register, o._register ) ; // add 'em
  return result; 
}

是去上班? 或者我需要声明的临时,

Vector3 add( const Vector3& o )
{
  __m256d d1 = *(__m256d*)(&x) ; // ? Cast to __m256d?
  __m256d d2 = *(__m256d*)(&o.x) ; // ? Cast to __m256d?
  __m256d result = _mm256_add_pd( d1, d2 ) ; // add 'em
  return Vector3( result ) ; // make a ctor that accepts __m256d?
}

编辑

我想出了这个例子,

#include <stdio.h>
#include <intrin.h>

int main()
{
  __m256d a, b, res;

  for( int i = 0; i < sizeof(__m256d)/sizeof(double); i++ )
  {
    a.m256d_f64[i] = i ;
    b.m256d_f64[i] = 2*i ;
  }

  // Perform __4__ adds.
  res = _mm256_add_pd(a, b);

  for( int i = 0; i < sizeof(__m256d)/sizeof(double); i++ )
  {
    printf("%f + %f = %f\n", a.m256d_f64[i], b.m256d_f64[i], res.m256d_f64[i]);
  }
  puts("");
}

我想现在的问题是, 是否 _mm256_add_pd 自动执行负载操作,或将事情搞的一团糟,如果我没有宣布我的__m256d使用它们的地方注册为当地人接近? (我怕的酒店房间/ deskdrawer型问题)

编辑2:

我尝试添加一个__m256寄存器我相当大的项目,我有一大堆

错误C2719:“值”:用__declspec(对齐('32' ))正式参数不对齐

错误,它使我相信,你不能__m256寄存器类内部,相反,他们应该被声明为本地人?

Answer 1:

首先,我想澄清一点混乱。 __m256d不是一个类型寄存器的,它是可以被加载到一个AVX寄存器的数据类型。 甲__m256d没有更多的寄存器比int是寄存器。 有几个方法可以得到进出的数据__m256d (或任何其他载体类型):

使用union是的, union招工作。 它工作得非常好,因为工会一般有正确的定位(尽管malloc可能没有,使用posix_memalign_aligned_malloc )。

class Vector3 {
public:
    Vector3(double xx, double yy, double zz);
    Vector3(__m256d vvec);


    Vector3 operator+(const Vector3 &other) const
    {
        return Vector3(_mm256_add_pd(vec, other.vec));
    }

    union {
        struct {
            double x, y, z;
        };
        __m256d vec; // a data field, maybe a register, maybe not
    };
};

使用内部函数:在函数里,它通常更容易使用内在进出载体类型来获取数据。

__m256d vec = ...;
double x, y, z;
vec = _mm256_add_pd(vec, _mm256_set_pd(x, y, z, 0.0));

使用指针转换:转换指针是有两个原因的最后手段。

  1. 指针可能无法正确对齐。

  2. 铸造指针有时编译器的别名分析一塌糊涂。

  3. 指针铸造绕过一些安全保障。

所以我只用指针铸造通过数据的大阵犁。



文章来源: Using __m256d registers