티스토리 뷰
안녕하세요. Teus입니다.
이번 포스팅은 SIMD(Single Instrunction Multi Data)를 다룹니다.
이번에는 AVX2의 기본 연산 중 쓸만한 함수 중 FMA연산을 정리합니다.
1. FMA
FMA는 Fused Multiply and Add 연산 입니다.
영단어에서 알 수 있듯
곱셈과 동시에 덧셈을 하는 연산입니다.
가장 간단하게 표현하면
ret = a*b + c
위 연산을 SIMD로 연산하는 명령어 입니다.
그럼 위 연산은 위해서 3개의 Vector가 필요하게 됩니다.
TMI
한번에 add와 mul연산을 동시에 할 수 있기 때문에, 한번에 2flops연산을 할 수가 있습니다.(한 cycle에 연산된다는 가정) 그래서 GPU에서는 core의 개수 * clock * 2flops로 성능이 측정됩니다. 그래서 현대 AI워크로드에 활용되는 GPU인 A100같은 경우 6912 * 1410 Mhz * 2Ops/hz = 19491840 => 19.5TFLOPs가 나오게 됩니다.(32bit Float 기준, 64bit Float기준으로는 절반이 됩니다)
Data Type | Description |
---|---|
_mm/mm256_fmadd_ps/pd/ | Multiply two vectors and add the product to a third (res = a * b + c) |
_mm/mm256_fmsub_ps/pd/ | Multiply two vectors and subtract a vector from the product (res = a * b - c) |
_mm_fmadd_ss/sd | Multiply and add the lowest element in the vectors (res[0] = a[0] * b[0] + c[0]) |
_mm_fmsub_ss/sd | Multiply and subtract the lowest element in the vectors (res[0] = a[0] * b[0] - c[0]) |
_mm/mm256_fnmadd_ps/pd | Multiply two vectors and add the negated product to a third (res = -(a * b) + c) |
_mm/mm256_fnmsub_ps/pd | Multiply two vectors and add the negated product to a third (res = -(a * b) - c) |
_mm_fnmadd_ss/sd | Multiply the two lowest elements and add the negated product to the lowest element of the third vector (res[0] = -(a[0] * b[0]) + c[0]) |
_mm_fnmsub_ss/sd | Multiply the lowest elements and subtract the lowest element of the third vector from the negated product (res[0] = -(a[0] * b[0]) - c[0]) |
_mm/mm256_fmaddsub_ps/pd | Multiply two vectors and alternately add and subtract from the product (res = a * b - c) |
_mm/mm256_fmsubadd_ps/pd | Multiply two vectors and alternately subtract and add from the product (res = a * b - c) |
FMA연산같은 경우 보면 알겠지만, DNN에 많이 사용되게 됩니다.
DNN이 output = input * weight + bias로 계산되기 때문이죵
이때 특이한 연산으로, 뒤에 ss/sd가 붙는 연산이 있습니다.
해당 연산같은 경우
3개 Vector의 첫번째 값에 대해서만 FMA연산을 하고
나머지 값은 a vector로 채우는 연산 입니다.
_mm_fmadd_ss
그래서 아래와 같이 코드를 실행하면, 첫 Vector의 값만 FMA를 시키고, 치환하는 효과를 보여줍니다.
(역시 비전공자 입장에서, 해당 연산이 어디에 사용되는지는 잘...)
#include <stdio.h>
#include <stdlib.h>
#include <malloc.h>
#include <xmmintrin.h>
#include <immintrin.h >
int main() {
__m128 arr1 = _mm_setr_ps(0.5, 0.9, 0.1, 0.4);
__m128 arr2 = _mm_setr_ps(0.1, 0.3, 0.7, 0.4);
__m128 arr3 = _mm_setr_ps(0.3, 0.2, 0.9, 0.5);
__m128 ret = _mm_fmadd_ss(arr1, arr2, arr3);
float* my_ret = (float*)&ret;
float* _arr1 = (float*)&arr1;
float* _arr2 = (float*)&arr2;
for (int i = 0; i < 4; i++) {
printf("idx : %d, arr1[%d] = %f, arr2[%d] = %f\n", i, i, _arr1[i], i, _arr2[i]);
}
for (int i = 0; i < 8; i++) {
printf("idx : %d, i : %f\n", i, my_ret[i]);
}
}
'C언어 잡기술 > SIMD(AVX)' 카테고리의 다른 글
AVX 튜토리얼6. 포인터 사용하기 (0) | 2024.04.24 |
---|---|
AVX 튜토리얼5. Permuting/Shuffling연산 (0) | 2024.04.24 |
AVX 튜토리얼3. mul/divide연산 (0) | 2024.04.24 |
AVX 튜토리얼2. Add/Sub연산 (0) | 2024.04.24 |
AVX 튜토리얼1. DataType (0) | 2024.04.02 |
- Total
- Today
- Yesterday
- Sort알고리즘
- Python
- 코딩테스트
- heap
- 알고리즘
- 완전탐색 알고리즘
- stack
- 동적계획법
- git
- Greedy알고리즘
- AVX
- 이분탐색
- C++
- SIMD
- GDC
- Search알고리즘
- 컴퓨터그래픽스
- prime number
- 프로그래머스
- 분할정복
- 병렬처리
- 자료구조
- hash
- 사칙연산
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | 5 | 6 | 7 |
8 | 9 | 10 | 11 | 12 | 13 | 14 |
15 | 16 | 17 | 18 | 19 | 20 | 21 |
22 | 23 | 24 | 25 | 26 | 27 | 28 |
29 | 30 | 31 |