AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / coding / Perguntas / 78010251
Accepted
MarkB
MarkB
Asked: 2024-02-17 06:17:14 +0800 CST2024-02-17 06:17:14 +0800 CST 2024-02-17 06:17:14 +0800 CST

Por que o GCC gera código que executa condicionalmente uma implementação SIMD?

  • 772

O código a seguir produz um assembly que executa condicionalmente o SIMD no GCC 12.3 quando compilado com -O3. Para completar, o código sempre executa SIMD no GCC 13.2 e nunca executa SIMD no clang 17.0.1.

#include <array>

__attribute__((noinline)) void fn(std::array<int, 4>& lhs, const std::array<int, 4>& rhs)
{
    for (std::size_t idx = 0; idx != 4; ++idx) {
        lhs[idx] = lhs[idx] + rhs[idx];
    }
}

Aqui está o link em godbolt.

Aqui está a montagem real do GCC 12.3 (com -O3):

fn(std::array<int, 4ul>&, std::array<int, 4ul> const&):
        lea     rdx, [rsi+4]
        mov     rax, rdi
        sub     rax, rdx
        cmp     rax, 8
        jbe     .L2
        movdqu  xmm0, XMMWORD PTR [rsi]
        movdqu  xmm1, XMMWORD PTR [rdi]
        paddd   xmm0, xmm1
        movups  XMMWORD PTR [rdi], xmm0
        ret
.L2:
        mov     eax, DWORD PTR [rsi]
        add     DWORD PTR [rdi], eax
        mov     eax, DWORD PTR [rsi+4]
        add     DWORD PTR [rdi+4], eax
        mov     eax, DWORD PTR [rsi+8]
        add     DWORD PTR [rdi+8], eax
        mov     eax, DWORD PTR [rsi+12]
        add     DWORD PTR [rdi+12], eax
        ret

Estou muito interessado em saber a) o propósito das primeiras 5 instruções de montagem eb) se há algo que possa ser feito para fazer com que o GCC 12.3 emita o código do GCC 13.2 (idealmente, sem escrever manualmente o SSE).

c++
  • 1 1 respostas
  • 591 Views

1 respostas

  • Voted
  1. Best Answer
    Peter Cordes
    2024-02-17T06:44:51+08:002024-02-17T06:44:51+08:00

    Parece que o GCC12 está tratando a classreferência como se fosse um simples int *, em termos de se lhse rhspoderia se sobrepor parcialmente .

    A sobreposição exata seria adequada, se lhs[idx]for o mesmo int que rhs[idx], lemos duas vezes antes de escrevê-lo. Mas com sobreposição parcial, rhs[3]por exemplo poderia ter sido atualizado por um dos lhs[0..2]acréscimos, o que não aconteceria com o SIMD se fizéssemos todas as cargas antes de qualquer uma das lojas.

    O GCC13 sabe que os objetos de classe não podem se sobrepor parcialmente (exceto para coisas comuns de sequência inicial para diferentes tipos de estrutura/classe, o que acho que não se aplica aqui). Isso seria UB, então podemos assumir que isso não acontece. A geração de código do GCC12 é uma otimização perdida.


    Então, como podemos ajudar o GCC12? A opção usual é __restrictremover verificações de sobreposição ou ativar a vetorização automática quando o compilador não deseja inventar verificações + um substituto. Em C, restrictfaz parte da linguagem, mas em C++ é apenas uma extensão. (Suportado pelos principais compiladores convencionais, e você pode usar o pré-processador #definepara a string vazia em outros.) Você pode usar __restrictreferências e também ponteiros. (Pelo menos o GCC e o Clang aceitam sem avisos -Wall; não verifiquei os documentos para ter certeza de que isso é padrão.)

    // downside: fn_restrict(same, same) would be UB
    void fn_restrict(std::array<int, 4>&__restrict lhs, const std::array<int, 4>& rhs)
    {
        for (std::size_t idx = 0; idx != 4; ++idx) {
            lhs[idx] = lhs[idx] + rhs[idx];
        }
    }
    

    Ou leia tudo manualmente lhsantes de escrever qualquer coisa

    Como o seu arrayé pequeno o suficiente para caber em um registro SIMD, não há ineficiência na cópia. Isso seria ruim array<int, 1000>ou algo assim!

    // downside: only efficient for small arrays that fit in a few vector regs at most
    void fn_temporary(std::array<int, 4>& lhs, const std::array<int, 4>& rhs)
    {
        auto sum = lhs;    // read the possibly-aliasing data into a temporary
        for (std::size_t idx = 0; idx != 4; ++idx) {
            sum[idx] += rhs[idx];  // update the temporary
        }
        lhs = sum;   // store back, after all loads
    }
    

    Ambos são compilados no mesmo conjunto autovetorizado do GCC13, sem instruções desperdiçadas ( Godbolt )

    # GCC12 -O3
    fn_temporary(std::array<int, 4ul>&, std::array<int, 4ul> const&):
            movdqu  xmm0, XMMWORD PTR [rsi]
            movdqu  xmm1, XMMWORD PTR [rdi]
            paddd   xmm0, xmm1
            movups  XMMWORD PTR [rdi], xmm0
            ret
    

    O alinhamento promissor (como alignas(16)um dos tipos?) Poderia permitir o uso de paddd xmm1, [rdi], um operando de origem de memória, sem AVX.

    • 16

relate perguntas

  • Por que os compiladores perdem a vetorização aqui?

  • Erro de compilação usando CMake com biblioteca [fechada]

  • Erro lançado toda vez que tento executar o premake

  • Como criar um tipo de octeto semelhante a std::byte em C++?

  • Somente operações bit a bit para std::byte em C++ 17?

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    Vue 3: Erro na criação "Identificador esperado, mas encontrado 'import'" [duplicado]

    • 1 respostas
  • Marko Smith

    Por que esse código Java simples e pequeno roda 30x mais rápido em todas as JVMs Graal, mas não em nenhuma JVM Oracle?

    • 1 respostas
  • Marko Smith

    Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores?

    • 1 respostas
  • Marko Smith

    Como faço para corrigir um erro MODULE_NOT_FOUND para um módulo que não importei manualmente?

    • 6 respostas
  • Marko Smith

    `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso?

    • 3 respostas
  • Marko Smith

    Quando devo usar um std::inplace_vector em vez de um std::vector?

    • 3 respostas
  • Marko Smith

    Um programa vazio que não faz nada em C++ precisa de um heap de 204 KB, mas não em C

    • 1 respostas
  • Marko Smith

    PowerBI atualmente quebrado com BigQuery: problema de driver Simba com atualização do Windows

    • 2 respostas
  • Marko Smith

    AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos

    • 1 respostas
  • Marko Smith

    Estou tentando fazer o jogo pacman usando apenas o módulo Turtle Random e Math

    • 1 respostas
  • Martin Hope
    Aleksandr Dubinsky Por que a correspondência de padrões com o switch no InetAddress falha com 'não cobre todos os valores de entrada possíveis'? 2024-12-23 06:56:21 +0800 CST
  • Martin Hope
    Phillip Borge Por que esse código Java simples e pequeno roda 30x mais rápido em todas as JVMs Graal, mas não em nenhuma JVM Oracle? 2024-12-12 20:46:46 +0800 CST
  • Martin Hope
    Oodini Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores? 2024-12-12 06:27:11 +0800 CST
  • Martin Hope
    sleeptightAnsiC `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso? 2024-11-09 07:18:53 +0800 CST
  • Martin Hope
    The Mad Gamer Quando devo usar um std::inplace_vector em vez de um std::vector? 2024-10-29 23:01:00 +0800 CST
  • Martin Hope
    Chad Feller O ponto e vírgula agora é opcional em condicionais bash com [[ .. ]] na versão 5.2? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench Por que um traço duplo (--) faz com que esta cláusula MariaDB seja avaliada como verdadeira? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng Por que `dict(id=1, **{'id': 2})` às vezes gera `KeyError: 'id'` em vez de um TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos 2024-03-20 03:12:31 +0800 CST
  • Martin Hope
    MarkB Por que o GCC gera código que executa condicionalmente uma implementação SIMD? 2024-02-17 06:17:14 +0800 CST

Hot tag

python javascript c++ c# java typescript sql reactjs html

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve